日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

【BZOJ2819】Nim 树状数组+LCA

發布時間:2025/3/15 编程问答 23 豆豆
生活随笔 收集整理的這篇文章主要介紹了 【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

Yes
No
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的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。