【BZOJ2819】Nim 树状数组+LCA
【BZOJ2819】Nim
Description
著名游戲設計師vfleaking,最近迷上了Nim。普通的Nim游戲為:兩個人進行游戲,N堆石子,每回合可以取其中某一堆的任意多個,可以取完,但不可以不取。誰不能取誰輸。這個游戲是有必勝策略的。于是vfleaking決定寫一個玩Nim游戲的平臺來坑玩家。
為了設計漂亮一點的初始局面,vfleaking用以下方式來找靈感:拿出很多石子,把它們聚成一堆一堆的,對每一堆編號1,2,3,4,...n,在堆與堆間連邊,沒有自環與重邊,從任意堆到任意堆都只有唯一一條路徑可到達。然后他不停地進行如下操作:
1.隨機選兩個堆v,u,詢問若在v到u間的路徑上的石子堆中玩Nim游戲,是否有必勝策略,如果有,vfleaking將會考慮將這些石子堆作為初始局面之一,用來坑玩家。
2.把堆v中的石子數變為k。
由于vfleaking太懶了,他懶得自己動手了。請寫個程序幫幫他吧。
Input
?第一行一個數n,表示有多少堆石子。
接下來的一行,第i個數表示第i堆里有多少石子。
接下來n-1行,每行兩個數v,u,代表v,u間有一條邊直接相連。
接下來一個數q,代表操作的個數。
接下來q行,每行開始有一個字符:
如果是Q,那么后面有兩個數v,u,詢問若在v到u間的路徑上的石子堆中玩Nim游戲,是否有必勝策略。
如果是C,那么后面有兩個數v,k,代表把堆v中的石子數變為k。
對于100%的數據:
1≤N≤500000, 1≤Q≤500000, 0≤任何時候每堆石子的個數≤32767
其中有30%的數據:
石子堆組成了一條鏈,這3個點會導致你DFS時爆棧(也許你不用DFS?)。其它的數據DFS目測不會爆。
注意:石子數的范圍是0到INT_MAX
Output
對于每個Q,輸出一行Yes或No,代表對詢問的回答。
Sample Input
【樣例輸入】5
1 3 5 2 5
1 5
3 5
2 5
1 4
6
Q 1 2
Q 3 5
C 3 7
Q 1 2
Q 2 4
Q 5 3
Sample Output
YesNo
Yes
Yes
Yes
題解:一個常識結論:Nim游戲先手必勝當且僅當所有堆的異或和不為0,否則先手必輸
然后用樹狀數組+倍增LCA維護DFS序的異或和就行了(當然,如果你維護的是入棧出棧序,可以不用倍增LCA)
#include <cstdio> #include <cstring> #include <iostream> using namespace std; const int maxn=500010; int n,m,cnt; int to[maxn<<1],next[maxn<<1],head[maxn],fa[maxn][20],dep[maxn],s[maxn],p[maxn],q[maxn],v[maxn]; char str[10]; int rd() {int ret=0,f=1; char gc=getchar();while(gc<'0'||gc>'9') {if(gc=='-')f=-f; gc=getchar();}while(gc>='0'&&gc<='9') ret=ret*10+gc-'0',gc=getchar();return ret*f; } void add(int a,int b) {to[cnt]=b,next[cnt]=head[a],head[a]=cnt++; } void dfs(int x) {p[x]=++p[0];for(int i=head[x];i!=-1;i=next[i])if(to[i]!=fa[x][0])fa[to[i]][0]=x,dep[to[i]]=dep[x]+1,dfs(to[i]);q[x]=p[0]; } void updata(int x,int val) {if(!x) return ;for(int i=x;i<=n;i+=i&-i) s[i]^=val; } int query(int x) {int i,ret=0;for(i=x;i;i-=i&-i) ret^=s[i];return ret; } int main() {scanf("%d",&n);int i,j,a,b,c,d;memset(head,-1,sizeof(head));for(i=1;i<=n;i++) v[i]=rd();for(i=1;i<n;i++) a=rd(),b=rd(),add(a,b),add(b,a);dep[1]=1,dfs(1);for(i=1;i<=n;i++) updata(p[i],v[i]),updata(q[i]+1,v[i]);for(j=1;(1<<j)<=n;j++)for(i=1;i<=n;i++) fa[i][j]=fa[fa[i][j-1]][j-1];m=rd();for(i=1;i<=m;i++){scanf("%s",str),a=rd(),b=rd();if(str[0]=='Q'){c=a,d=b;if(dep[a]<dep[b]) swap(a,b);for(j=19;j>=0;j--) if(dep[fa[a][j]]>=dep[b]) a=fa[a][j];if(a!=b){ for(j=19;j>=0;j--) if(fa[a][j]!=fa[b][j]) a=fa[a][j],b=fa[b][j];a=fa[a][0];}if(query(p[c])^query(p[d])^v[a]) printf("Yes\n");else printf("No\n");}if(str[0]=='C'){updata(p[a],v[a]),updata(q[a]+1,v[a]);updata(p[a],b),updata(q[a]+1,b);v[a]=b;}}return 0; }轉載于:https://www.cnblogs.com/CQzhangyu/p/7044261.html
總結
以上是生活随笔為你收集整理的【BZOJ2819】Nim 树状数组+LCA的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: wmi接口如何通过Win32_Volum
- 下一篇: ACdream 1431 Sum vs