最大异或对(字典树)

字典树可以看一下这位大佬的文章:

【数据结构】字典树TrieTree图文详解_Avalon Demerzel的博客-CSDN博客

4、最大异或对

给定N个整数A1、A2、A3……A**N;从中选出两个数进行异或运算,得到结果的最大值是多少?

数据范围:

1≤N≤10^5

1≤A(i)≤10^8;

Input

第一行输入一个数N

第二行输入N个数表示:A1、A2……A(N)

Output

输出一个数,表示答案。

Sample Input 1

3
1 2 3

Sample Output 1

3

Source

#include<stdio.h>
#include<assert.h>
#include<malloc.h>
#include<math.h>
#include<string.h>
int son[100010][2] = { 0 };
int idx = 1;
int a[100010] = { 0 };

int max(a, b)
{
	if (a > b)
		return a;
	else
		return b;
}

void insert(int x)
{
	int p = 0;
	int i = 0;
	for (i = 30; i >= 0; --i)
	{
        //取出x的二进制最高位
		int u = x >> i & 1;
		if (son[p][u] == 0)
		{
			son[p][u] = idx;
			++idx;
		}
		p = son[p][u];
	}
}

//用于找寻某个数在已建字典树中
//对应最大异或数是多少
int find(int x)
{
	int p = 0, res = 0;
	int i = 0;
	for (i = 30; i >= 0; --i)
	{
		int u = x >> i & 1;
		//检查对应层有无不同的数
		if (son[p][!u])	//有不同数
		{
			p = son[p][!u];	//p就指到不同数的地址
            //res=res+(1<<i);
            //还原某数最大异或数对应二进制数字为十进制
            //原理:难以理解┭┮﹏┭┮(原理见下图)
            
            //原理:补充,二进制本质为某个数进行短除法的余数
            //而res = (res * 2) + !u是对短除的逆过程
			res = (res * 2) + !u;
		}
		else  //没有不同数
		{
			p = son[p][u];
			res = (res * 2) + u;
		}
	}
	return res;
}

int main()
{
	int n = 0;
	scanf("%d", &n);
	for (int i = 0; i < n; i++)  
		scanf("%d", &a[i]);

	int res = 0;
	for (int i = 0; i < n; i++)
	{
		insert(a[i]);
		int t = find(a[i]);
		res = max(res, a[i] ^ t);
	}
	printf("%d\n", res);
	return 0;
}

 

 

 

  • 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、付费专栏及课程。

余额充值