分析
考虑把边权转为点权。设每个点的点权为所有与它相连的边的边权的异或和。
那么可以发现,$u$ 到 $v$ 的路径上的边权异或 $w$,相当于 $u$ 和 $v$ 的点权异或一个 $w$。
于是问题变成给你一些数,每次可以选出两个数并让它们异或一个数,求让所有数变为 $0$ 的最小操作次数。
显然可以先贪心地去消,把所有相同的先消掉。这样子会剩下至多 $15$ 个互不相同的数。
于是可以考虑状压 DP。设 $f_S$ 表示 $S$ 中的数全部变为 $0$ 的最小操作次数,边界是 $f_S = \operatorname{popcount}(S) - 1$,枚举合法子集转移即可。
因为一次操作不改变所有数的异或和,所以一个集合合法当且仅当其中所有数异或和为 $0$。
时间复杂度 $\mathcal{O}(n\log n + 3^{15})$。
代码
//It is made by M_sea
#include <algorithm>
#include <iostream>
#include <cstdlib>
#include <cstring>
#include <cstdio>
#include <cmath>
#define re register
using namespace std;
inline int read() {
int X=0,w=1; char c=getchar();
while (c<'0'||c>'9') { if (c=='-') w=-1; c=getchar(); }
while (c>='0'&&c<='9') X=X*10+c-'0',c=getchar();
return X*w;
}
const int N=200000+10;
int w[N],cnt[20],f[1<<16],sxor[1<<16];
int main() {
int n=read();
for (re int i=1;i<n;++i) {
int u=read()+1,v=read()+1,val=read();
w[u]^=val,w[v]^=val;
}
for (re int i=1;i<=n;++i) ++cnt[w[i]];
int ans=0,s=0;
for (re int i=1;i<=15;++i) ans+=cnt[i]/2,s|=(cnt[i]&1)<<(i-1);
for (re int i=1;i<(1<<15);++i) f[i]=f[i>>1]+(i&1);
for (re int i=1;i<(1<<15);++i) --f[i];
for (re int i=1;i<(1<<15);++i)
for (re int j=1;j<=15;++j)
if (i&(1<<(j-1))) sxor[i]^=j;
for (re int i=1;i<(1<<15);++i) {
if (sxor[i]) continue;
for (re int j=i&(i-1);j;j=(j-1)&i)
if (!sxor[j]) f[i]=min(f[i],f[j]+f[i^j]);
}
printf("%d\n",ans+f[s]);
return 0;
}