zoukankan      html  css  js  c++  java
  • HDU6625: three arrays (字典树处理xor)

    题意:给出A数组,B数组,你可以对A和B分别进行重排列,使得C[i]=A[i]^B[i]的字典序最小。

    思路:对于这类题,显然需要建立字典树,然后某种形式取分治,或者贪心。  假设现在有了两颗字典树A,B,显然尽量让同方向的先匹配。

    而且同一棵树的左右两边相互不影响,所以可以直接贪:如果A树上出发左走有x个数字,B左走有y个数字,那么一定会左匹配min(x,y);右匹配同理; 剩下的交叉匹配;  

    看代码应该就会看懂了:add建立字典树。

                                            dfs进行匹配;  (i,j)从前往后分别是,(0,0) (1,1),(0,1) (1,0)保证了相同的先匹配。

    #include<bits/stdc++.h>
    #define rep(i,a,b) for(int i=a;i<=b;i++)
    using namespace std;
    const int maxn=100010;
    int ch0[maxn*30][2],ch1[maxn*30][2],tot0,tot1;
    int ans[maxn],num,sum0[maxn*30],sum1[maxn*30];
    void init()
    {
        rep(i,0,tot0) rep(j,0,1) ch0[i][j]=0;
        rep(i,0,tot1) rep(j,0,1) ch1[i][j]=0;
        tot0=tot1=num=0;
    }
    void add(int ch[][2],int sum[],int x,int &tot)
    {
        for(int i=29,now=0;i>=0;i--){
            int t=(x>>i)&1;
            if(!ch[now][t]) ch[now][t]=++tot;
            now=ch[now][t];
            sum[now]++;
        }
    }
    void dfs(int now0,int now1,int cost,int dep)
    {
        int e=min(sum0[now0],sum1[now1]);
        sum0[now0]-=e; sum1[now1]-=e;
        if(dep==-1) { 
            rep(i,1,e) ans[++num]=cost;
            return ;
        }
        rep(k,1,4){
            int i=k&1,j=i^1; if(k<=2) j=i;
            if(sum0[ch0[now0][i]]&&sum1[ch1[now1][j]])
              dfs(ch0[now0][i],ch1[now1][j],cost+(i==j?0:(1<<dep)),dep-1);
        }
    }
    int main()
    {
        int T,N,x;
        scanf("%d",&T);
        while(T--){
            init();
            scanf("%d",&N);
            rep(i,1,N){
                scanf("%d",&x);
                add(ch0,sum0,x,tot0);
            }
            rep(i,1,N){
                scanf("%d",&x);
                add(ch1,sum1,x,tot1);
            }
            dfs(0,0,0,29);
            sort(ans+1,ans+N+1);
            rep(i,1,N-1) printf("%d ",ans[i]);
            printf("%d
    ",ans[N]);
        }
        return 0;
    }
  • 相关阅读:
    hdoj--2098--分拆素数和(枚举)
    hdoj--3594--Cactus(tarjan)
    hdoj--1251--统计难题(字典树)
    hdoj--2534--Score(gcd)
    nyoj--1185--最大最小值(线段树)
    hdoj--1166--敌兵布阵(线段树)
    hdoj--1754--I Hate It(线段树)
    poj--2234--Matches Game(尼姆博弈)
    lightoj--1005--Rooks(组合数)
    SPOJ
  • 原文地址:https://www.cnblogs.com/hua-dong/p/11310561.html
Copyright © 2011-2022 走看看