COJ 1216: 異或最大值(01字典樹)


題意:求n個非負數中任意2個的異或值的最大值。n數量級為10^5

分析:直接暴力肯定超時了。一個非負整數可以看成1個32位的01字符串,n個數可以看成n個字符串,因此可以建立字典樹,建好樹后,對於任意非負整數x,可以沿着樹根往下貪心找到y,使得x異或y最大,復雜度為樹的深度。

View Code
#include <stdio.h>
#include <string.h>
#define MAX(a,b) ((a)>(b)?(a):(b))
#define NODE 3200010
#define N 100010
int n;
int v[N];
int node;
int next[NODE][2];
int end[NODE];
void add(int cur,int k)
{
    memset(next[node],0,sizeof(next[node]));
    end[node]=0;
    next[cur][k]=node++;
}
int cal(int x)
{
    int i,k,cur=0;
    for(i=30;i>=0;i--)
    {
        k=((1<<i)&x)?0:1;
        if(next[cur][k]) cur=next[cur][k];
        else    cur=next[cur][1-k];
    }
    return (x^end[cur]);
}
int main()
{
    int i,j,k,x,cur;
    int ans;
    while(~scanf("%d",&n))
    {
        node=1;
        memset(next[0],0,sizeof(next[0]));
        for(i=0;i<n;i++)
        {
            scanf("%d",&x);
            v[i]=x;
            cur=0;
            for(j=30;j>=0;j--)
            {
                k=((1<<j)&x)?1:0;
                if(next[cur][k]==0) add(cur,k);
                cur=next[cur][k];
            }
            end[cur]=x;
        }
        for(ans=i=0;i<n;i++)    ans=MAX(ans,cal(v[i]));
        printf("%d\n",ans);
    }
    return 0;
}

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM