HDU 5465 Clarke and puzzle Nim游戏+二维树状数组

时间:2023-03-09 02:32:43
HDU 5465 Clarke and puzzle Nim游戏+二维树状数组

题目链接:

http://acm.hdu.edu.cn/showproblem.php?pid=5465

Clarke and puzzle

 Accepts: 42
 Submissions: 269
 Time Limit: 4000/2000 MS (Java/Others)
 Memory Limit: 65536/65536 K (Java/Others)
问题描述
克拉克是一名人格分裂患者。某一天,有两个克拉克(aa和bb)在玩一个方格游戏。
这个方格是一个n*mn∗m的矩阵,每个格子里有一个数c_{i, j}c​i,j​​。
aa想开挂,想知道如何打败bb。
他们要玩qq次游戏,每一次做一次操作:
1. 取出当中的一个子矩阵(x_1, y_1)-(x_2, y_2)(x​1​​,y​1​​)−(x​2​​,y​2​​)玩游戏。两个人轮流行动,每一次只能从这个子矩阵中的一个方格c_{i, j}c​i,j​​中减掉一个的数d(1 \le d \le c_{i, j})d(1≤d≤c​i,j​​),当一个格子的数为00时则不能减。如果操作完后另一者无法操作,那么胜利。否则失败。现在aa作为先手,想知道是否存在一种方案使得自己胜利。
2. 将c_{i, j}c​i,j​​的数改成bb
输入描述
第一行一个整数T(1 \le T \le 5)T(1≤T≤5),表示数据的组数。
每组数据第一行为三个整数n, m, q(1 \le n, m \le 500, 1 \le q \le 2*10^5)n,m,q(1≤n,m≤500,1≤q≤2∗10​5​​)。
接下来是一个nn行mm列的矩阵,其中第ii行第jj列的数为c_{i, j}(0 \le c_{i, j} \le 10^9)c​i,j​​(0≤c​i,j​​≤10​9​​)。
接下来时qq行,第一个数为optopt。当opt=1opt=1时,后面接着四个整数,依次表示x_1, y_1, x_2, y_2(1 \le x_1 \le x_2 \le n, 1 \le y_1 \le y_2 \le m)x​1​​,y​1​​,x​2​​,y​2​​(1≤x​1​​≤x​2​​≤n,1≤y​1​​≤y​2​​≤m),表示一个询问;当opt=2opt=2时,后面接着三个整数x, y, z(1 \le x \le n, 1 \le y \le m, 0 \le z \le 10^9)x,y,z(1≤x≤n,1≤y≤m,0≤z≤10​9​​),表示将c_{x, y}c​x,y​​更改为zz。
输出描述
对于每组数据,每个询问输出aa是否能胜利,如果能,输出YesYes,否则输出NoNo。  
输入样例
1
1 2 3
1 2
1 1 1 1 2
2 1 2 1
1 1 1 1 2
输出样例
Yes
No
Hint
第一个询问:一开始aa可以在(1, 2)(1,2)的格子上减掉11,则接下来无论bb怎么选,都还剩一个11,所以aa胜利。
第二个询问:无论aa怎么选,都还剩下一个11,所以bb胜利。

题解:

  Nim游戏有个结论:只要所有的数异或为零,则先手必败。因此我们只要用二维树状数组来维护二维区间的所有数的异或值就可以了。

代码:

 #include<iostream>
#include<cstdio>
#include<cstring>
using namespace std; typedef long long LL; const int maxn=; int lowbit(int x){
return x&-x;
} int n,m,q;
int a[maxn][maxn];
int c[maxn][maxn]; void update(int x,int y,int v){
for(int i=x;i<=n;i+=lowbit(i)){
for(int j=y;j<=m;j+=lowbit(j)){
c[i][j]^=v;
}
}
} int query(int x,int y){
int ret=;
for(int i=x;i>;i-=lowbit(i)){
for(int j=y;j>;j-=lowbit(j)){
ret^=c[i][j];
}
}
return ret;
} void init(){
memset(c,,sizeof(c));
} int main(){
int tc;
scanf("%d",&tc);
while(tc--){
init();
scanf("%d%d%d",&n,&m,&q); for(int i=;i<=n;i++){
for(int j=;j<=m;j++){
int v;
scanf("%d",&v);
a[i][j]=v;
update(i,j,v);
}
}
int op;
while(q--){
scanf("%d",&op);
if(op==){
int x1,y1,x2,y2;
scanf("%d%d%d%d",&x1,&y1,&x2,&y2);
int ans=query(x2,y2)^query(x1-,y2)^query(x2,y1-)^query(x1-,y1-);
//两次异或相当于减,所以这里相当于是:query(x2,y2)-query(x1-1,y2)-query(x2,y1-1)+query(x1-1,y1-1),容斥的思想;
if(ans) printf("Yes\n");
else printf("No\n");
}else{
int x,y,v;
scanf("%d%d%d",&x,&y,&v);
update(x,y,v^a[x][y]);
a[x][y]=v;
}
}
}
return ;
}