AcWing 143 最大异或对 题解(字典树求树上异或和 模板)( 字典树 Tire)

异或运算^:相同取1, 不同取0

#include<iostream>
#include<algorithm>

using namespace std;

const int N = 1e5 + 10;

int son[N * 31][2], idx;//son[]记录线段树可能有1e5个31位的二进制树,所以son[31*1e5][2] 
int n;
int a[N];

void insert(int x){
	int p = 0;//字典树根节点
	for(int i = 30; i >= 0; i -- ){//对于这个二进制数,从高位开始往树内存 
		int u = x >> i & 1;//取出x第i位的数(1或0)
		if(!son[p][u]) son[p][u] = ++ idx;//如果这个节点,创建这个节点,节点下标为++idx;
		p = son[p][u];//继续向下遍历
	} 
}

int query(int x){
	int res = 0, p = 0;//p从根节点开始遍历,res储存结果 
	for(int i = 30; i >= 0; i -- ){
		int u = x >> i & 1;//取出x的第i位数字(1或0)
		if(son[p][!u]){//异或和运算:两数不同结果为1,所以要找和u相反的节点 
			res = res * 2 + !u;//如果这个点存在,就让res左移一位(res此时为二进制数),末尾加上!u
			p = son[p][!u];//继续向下遍历,因为每一位都要进行异或运算,所以要沿着和x进行异或运算结果最大的树的数向下遍历
		} 
		else{//如果和!u不存在,则只能走u节点 
			res = res * 2 + u;//res左移一位,末尾加上u
			p = son[p][u];//继续向下遍历 
		} 
	} 
	return res;
} 

int main()
{
	scanf("%d", &n);
	for(int i = 0; i < n; i ++ ){
		scanf("%d", &a[i]);
		insert(a[i]);
	}
	int rv = 0;
	for(int i = 0; i < n; i ++ ){
		int t = query(a[i]);//令t等于和a[i]异或和最大的数
		rv = max(rv, t^a[i]);//遍历选出最大值 
	}
	printf("%d", rv);
	return 0; 
}
  • 0
    点赞
  • 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、付费专栏及课程。

余额充值