Trie(最大异或对)

什么是异或_异或运算及异或运算的作用_cxu123321的博客-CSDN博客_异或

题目:

暴力做法(会超时):

先固定其中一个数Ai,然后从A1~An中选出一个值Aj,使得Ai^Aj最大,枚举Ai

#include<iostream>
using namespace std;
const int N=10e5+10;
int n,a[N];
int main(){
    cin>>n;
    for(int i=0;i<n;i++)cin>>a[i];
    int res=0;
    for(int i=0;i<n;i++){
        for(int j=0;j<n;j++){
            res=max(res,a[i]^a[j]);
        }
    }
    cout<<res;
    return 0;
}

优化:trie 

代码: 

#include<iostream>
#include<algorithm>
using namespace std;

int const N=100010,M=31*N;
int n;
int a[N];
int son[M][2],idx;                    //M代表一个数字串二进制可以到多长

void insert(int x)
{
    int p=0;                         //根节点
    for(int i=30;~i;i--)             //从最高位开始枚举,i>=0等价于~i
    {
        int &s=son[p][x >> i & 1];   //看一下x的二进制表示中第i位是0还是1
        if(!s) s = ++ idx;           //如果插入中发现没有该子节点,就创建新节点,idx是下一个点的下标
        p = s;                       //p走到s的位置,指针指向下一层
    }
}

int query(int x)                   //把每一个整数看出一个31位长度的二进制数
{
    int p=0;int res=0;              //p是在树上遍历的指针
    for(int i=30;i>=0;i--)
    {                               
        int s = x >> i & 1;
        if(son[p][!s])              //和当前不一样的分支存在时
        {  
          p=son[p][!s];             //p指针就走到这个不一样的分支上
          res=res*2+1;              //等价于res+=1<<i;             
        }                                                                                                                                                  
        
        else                        //否则就往当前分支走
        {
            p=son[p][s];
            res=res*2+0;            //等价于res+=0<<i;此步可以省略
        }
    }
    return res;
}

int main(void)
{
    cin>>n;
    for(int i=0;i<n;i++)
    {
        cin>>a[i];
        insert(a[i]);  //建索引
    }
    int res=0;
    for(int i=0;i<n;i++)
    {   
        res=max(res,query(a[i]));  //query(a[i])查找的与当前这个数异或值最大的结果
    }
    cout<<res<<endl;
}

 

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
求1~n数组的最大异或和也可以使用类似于求一段区间最大异或和的方法。具体步骤如下: 1. 将1~n数组中的所有数以二进制形式插入到字典树中。 2. 对于每个数,从高位到低位依次匹配字典树上的节点,如果当前位为1,就往字典树的右子树走,否则就往左子树走。匹配完整个二进制数后,我们可以得到一个最大异或值。 3. 对于1~n数组,我们可以将其中的相邻两个数看作一段区间,然后使用类似于求一段区间最大异或和的方法求出最大异或和。 时间复杂度为O(nlogC),其中n为数组长度,C为数的范围。以下是求1~n数组的最大异或和的C++代码: ```c++ #include <iostream> using namespace std; const int MAXN = 100010; const int MAXBITS = 30; struct TrieNode { int cnt; int children[2]; } trie[MAXN * MAXBITS]; int root, node_cnt; void insert(int x) { int p = root; for (int i = MAXBITS - 1; i >= 0; i--) { int idx = (x >> i) & 1; if (!trie[p].children[idx]) { trie[p].children[idx] = ++node_cnt; } p = trie[p].children[idx]; trie[p].cnt++; } } int query(int x) { int p = root, res = 0; for (int i = MAXBITS - 1; i >= 0; i--) { int idx = (x >> i) & 1; if (trie[trie[p].children[idx ^ 1]].cnt > 0) { res += (1 << i); p = trie[p].children[idx ^ 1]; } else { p = trie[p].children[idx]; } } return res; } int main() { int n; cin >> n; root = 1; node_cnt = 1; int pre = 0, ans = 0; for (int i = 1; i <= n; i++) { int x; cin >> x; insert(pre); pre ^= x; ans = max(ans, query(pre)); } cout << ans << endl; return 0; } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值