AT3913 XOR Tree

AT3913 XOR Tree

题意

就是给你一棵树,树上的每一条边有一个权值( 0 ∼ 15 0 \sim 15 015)
每次可以选树上的一条路径,把路径上的每一条边都异或上一个值
问最少几次操作可以使所有边的边权都变为0

题解

首先是一个非常套路巧妙的问题转化

设每个点的权值为和它相邻的边的边权的异或和

发现要使边权的值为0的充要条件是所有点的点权都为0

具体的证明也十分简单,可以利用它是一棵树来证明

然后问题就转化为了给你若干 0 ∼ 15 0 \sim 15 015的数,然后每次可以选两个数然后让它们异或上一个值。
相同的数可以直接取掉,0可以直接扔掉

然后剩下一些两两不同的数,直接状压DP即可

code:

#include<bits/stdc++.h>
#define N 200005
using namespace std;
int bitcount(int S) {
	int ret = 0;
	while(S) ret ++, S -= (S & (-S));
	return ret;
}
int check(int S) {
	int ret = 0;
	for(int i = 0; i < 15; i ++) if(S & (1 << i)) ret ^= (i + 1);
	return ret;
}
int n, d[N], cnt[N], f[N];
int main() {
	scanf("%d", &n);
	for(int i = 1; i < n; i ++) {
		int u, v, c;
		scanf("%d%d%d", &u, &v, &c);
		d[u] ^= c, d[v] ^= c;
	}
	for(int i = 0; i < n; i ++) cnt[d[i]] ++;
	int ans = 0, sta = 0;
	for(int i = 1; i < 16; i ++) ans += cnt[i] / 2, sta |= (cnt[i]&1) << (i - 1);
	memset(f, 0x3f, sizeof f);
	f[0] = 0;
	for(int S = 1; S < (1 << 15); S ++) { 
		if((sta & S) != S || check(S)) continue;
		f[S] = bitcount(S) - 1;
		for(int SS = S; SS; SS = (SS - 1) & S) if(!check(SS)){ 
			f[S] = min(f[S], f[SS] + f[S ^ SS]);
		}
	}
	int ret = 0;
	printf("%d", ans + f[sta]);
	return 0;
} 

总结

问题转换十分重要

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值