AtCoder

分析

考虑把边权转为点权。设每个点的点权为所有与它相连的边的边权的异或和。

那么可以发现,$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;
}
最后修改:2021 年 03 月 23 日 04 : 41 PM