最长异或路径
最长异或路径
题目描述
给定一棵\(n\)个点的带权树,结点下标从\(1\)开始到\(N\)。寻找树中找两个结点,求最长的异或路径。 异或路径指的是指两个结点之间唯一路径上的所有边权的异或。
输入输出格式
输入格式
第一行一个整数\(N\),表示点数。 接下来 \(n-1\) 行,给出 \(u,v,w\) ,分别表示树上的 \(u\) 点和 \(v\) 点有连边,边的权值是 \(w\)。
输出格式
一行,一个整数表示答案。
输入输出样例
输入样例 #1
4
1 2 3
2 3 4
2 4 6
输出样例 #1
7
说明
最长异或序列是\(1-2-3\),答案是 \(7 (=3 ⊕ 4)\)
数据范围 \(1\le n \le 100000;0 < u,v \le n;0 \le w < 2^{31}\)
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<queue>
#include<cmath>
using namespace std;
struct node{
int v;
int w;
int nxt;
}edge[2000001];
int head[2000001];
int cnt=-1;
void add(int u,int v,int w){
edge[++cnt].nxt=head[u];
edge[cnt].v=v;
edge[cnt].w=w;
head[u]=cnt;
}
int sum[2000001];
void dfs(int x,int fa){
for(int i=head[x];~i;i=edge[i].nxt){
int v=edge[i].v;
int w=edge[i].w;
if(v!=fa){
sum[v]=sum[x]^w;
dfs(v,x);
}
}
}
struct trie{
int ch[2];
}t[2000001];
int tot;
void build(int val,int x){
for(int i=(1<<30);i;i>>=1){
bool c=val&i;
if(!t[x].ch[c]){
t[x].ch[c]=++tot;
}
x=t[x].ch[c];
}
}
int query(int val,int x){
int ans=0;
for(int i=(1<<30);i;i>>=1){
bool c=val&i;
if(t[x].ch[!c]){
ans+=i;
x=t[x].ch[!c];
}
else x=t[x].ch[c];
}
return ans;
}
int main(){
freopen("in.in","r",stdin);
freopen("out.out","w",stdout);
memset(head,-1,sizeof(head));
int n;
scanf("%d",&n);
for(int i=1;i<n;++i){
int u,v,w;
scanf("%d%d%d",&u,&v,&w);
add(u,v,w);
add(v,u,w);
}
dfs(1,-1);
for(int i=1;i<=n;++i){
build(sum[i],0);
}
int ans=0;
for(int i=1;i<=n;++i){
ans=max(ans,query(sum[i],0));
}
printf("%d\n",ans);
}