xsy1531[北京集训2016] 魔法游戏
orz sjk
题目大意
有一棵树,两个人每个节点上有一个权值,两个人轮流选择一个根节点将其权值\(k\)除以\([2,k+1]\)若除到\(0\)就删去此根节点,它的儿子变成新根节点,删掉最后一个点赢。求先手还是后手必胜。
题解
首先考虑一些只有一个根节点的树,如果以二进制角度看,每次除以\([2,k+1]\)的数相当于拿掉一些二进制位的1,至少拿掉顶上的,可以拿完,这就相当于nim取石头游戏。每个权值相当于有\(log_2k+1\)个石头。
那么设\(SG(u,i)\)表示节点\(u\)有\(i\)个石头的\(SG\)值,显然叶子节点\(u\)的\(SG(u,i)=i\),非叶子节点\(u\)的\(SG(u,0)=SG(son_1(u),k_1(u))\ XOR\ SG(son_2(u),k_2(u))\ XOR...\) 根据\(SG(u,k)=mex\{ SG(u,k)的后继状态 \}\) 那么 \(SG(u,k)=mex \{ SG(u,0),SG(u,1)...SG(u,k-1)\}\) 本来\(SG(u,0)=0\)那么\(SG(u,k)=k\)现在\(SG(u,0)\)等于一个不为零的数\(x\)那么\(SG(u,1)=0\ SG(u,2)=1\ ... SG(u,x)=x-1\)直到\(SG(u,x+1)=x+1\).
#include<bits/stdc++.h>
using namespace std;
typedef unsigned long long ll;
const int N=100010;
struct E{
int t,xt;
}e[N<<1];
int h[N],cnt;
void adge(int f,int t){
e[++cnt]=(E){t,h[f]};h[f]=cnt;
}
ll mi[100] ,val[N];
int sg[N];
void dfs(int u,int fa){
sg[u]=0;
int x=upper_bound(mi,mi+64,val[u])-mi;
bool yz=1;
for(int i=h[u];~i;i=e[i].xt) if(e[i].t!=fa){
yz=0;
dfs(e[i].t,u);
sg[u]^=sg[e[i].t];
}
if(!yz) sg[u]=x-(x<=sg[u]);
else sg[u]=x;
}
int main(){
memset(h,cnt=-1,sizeof(h));
mi[0]=1;
for(int i=1;i<=63;i++) mi[i]=mi[i-1]*2;
int n;
while(scanf("%d",&n)!=EOF){
for(int i=0;i<n;i++) scanf("%llu",&val[i]);
for(int i=1;i<n;i++){
int a,b;
scanf("%d%d",&a,&b);
adge(a,b);
adge(b,a);
}
dfs(0,0);
puts(sg[0]?"Alice":"Marisa");
for(int i=0;i<n;i++) h[i]=-1;
cnt=-1;
}
return 0;
}