【AtCoder Regular Contest 083 E】Bichrome Tree
贪心,构造。
题目大意:
给定一棵 $n$ 个结点的有根数,每个结点有一个数值 $X_i$。
现在要求把结点染成黑白两种颜色,并对每个结点指定一个权值 $a_i$。
要求满足:$i$ 的子树中和 $i$ 同色结点的权值和恰好是 $X_i$.
问是否有合法方案。
题解:
考虑从下往上依次确定。
确定一个结点的权值时,它的每个儿子结点都有两种不同的权值可以使用。我们无需管它到底是黑点还是白点,因为把整棵子树颜色翻转后还是合法。
对于一棵子树,其中一个权值一定是根结点。我们希望另外一个权值尽可能小,这样才能匹配更多的权值。
在每层处做一个背包问题即可得出所有可能的不包括根的子树和。
我们要传上去的尽可能小,就是要使花费的尽可能大,所以找一个比 $X_i$ 小,且能得到的权值和 $v$,令 $a_i=X_i-v$ 即可。
如果某次找不到,则说明无解。
Code:
#include<cstdio>
#include<bitset>
#include<vector>
#include<cstdlib>
using namespace std;
const int N=1145;
int n,fa[N],X[N],a[N];
vector<int>G[N];
bitset<5001>b;
void dfs(int now){
int sum=0;
for(int to:G[now])dfs(to),sum+=a[to];
b.reset();
b.set(0);
for(int to:G[now])b=(b<<X[to])|(b<<(a[to]-X[to]));
int x=-1;
for(int i=X[now];~i&&x==-1;--i)if(b.test(i))x=i;
if(x==-1)exit(puts("IMPOSSIBLE")*0);
a[now]=sum+X[now]-x;
}
int main(){
scanf("%d",&n);
for(int i=2;i<=n;++i)
scanf("%d",fa+i),G[fa[i]].push_back(i);
for(int i=1;i<=n;++i)scanf("%d",X+i);
dfs(1);
puts("POSSIBLE");
return 0;
}