Tree Constructer 构造 + 二分图
题目大意:
给你一棵树,让你去重新赋予每一个节点一个权值,如果节点 \((u,v)\) 连边,那么 \((a_u \,or\,a_v )= 2^{60}-1\) ,反之也成立,问你如何进行赋值,保证最后连成的树是题目所给定的。
题解:
比赛的时候,想的很慌乱,没有想到树本来就是一个二分图。
- 树是一个二分图,那么变成两个点集,因为树最多只有100个节点,那么这个点集里面肯定有一个的节点数量小于等于50
- 所以我就取点集小的设为 A,另一个设为B,因为要求等于 \(2^{60}-1\) ,那么就一共有60位可以分配,首先保证两个点集内部不想连,所以最高位分配,A最高位为0,B最高位为1。
- 接下来就主要操作A点集,A点集的每一个数我都给它分配一位,如果A点集有 \(x\) 位,那么就分配了 \(x\) 位了,就相当于A点集每个节点是一把锁,每一位就是一把不同的锁,B点集就是钥匙,它和谁连就需要谁的钥匙,那位就置为1,其他不需要的就置为0
非常有趣的一个构造,可惜没有想出来。。。
#include <bits/stdc++.h>
#define inf 0x3f3f3f3f
using namespace std;
typedef long long ll;
const int maxn = 2e2+10;
vector<int>G[maxn];
void add(int u,int v){
G[u].push_back(v);
G[v].push_back(u);
}
int dep[maxn],num[2];
void dfs(int u,int pre,int d){
dep[u] = d,num[d&1]++;
for(int i=0;i<G[u].size();i++){
int v = G[u][i];
if(v == pre) continue;
dfs(v,u,d+1);
}
}
ll bit[maxn];
int main(){
int n;
scanf("%d",&n);
for(int i=1,u,v;i<n;i++){
scanf("%d%d",&u,&v);
add(u,v);
}
dfs(1,0,1);
int x = 0;
if(num[1]<num[x]) x = 1;
ll S = 1ll<<59,now = 1;
for(int i=1;i<=n;i++){
if((dep[i]&1)==x){
bit[i] = (S - 1)^now;
now = now*2;
}
}
for(int i=1;i<=n;i++){
if((dep[i]&1)!=x){
bit[i] = S;
for(int j=0;j<G[i].size();j++){
int v = G[i][j];
bit[i] |= bit[v]^(S-1);
}
}
}
for(int i=1;i<=n;i++) {
printf("%lld",bit[i]);
if(i==n) printf("\n");
else printf(" ");
}
return 0;
}