BZOJ 3729 - Gty的游戏(Staircase 博弈+时间轴分块)
介于自己以前既没有写过 Staircase-Nim 的题解,也没写过时间轴分块的题解,所以现在就来写一篇吧(fog
首先考虑最极端的情况,如果图是一条链,并且链的一个端点是 \(1\),那么问题可以转化为序列上的问题,即可视作有一个 \(n\) 级的阶梯,第 \(i\) 级台阶上有 \(a_i\) 个石子,每次可以选择一堆并将其中 \(x(x\le L)\) 个石子向下移动一级,不能移动最底下一级的石子,不能操作者输。
这是一个经典的 staircase 博弈的模型,对于这样的模型我们有一个结论:我们并不用关心奇数级台阶上的石子的情况,只用把偶数级台阶上石子的 SG 值异或起来就是最终游戏的异或值。因为如果先手有必胜策略那么一旦后手将奇数级台阶上的石子移动到了偶数级台阶上,那么先手一定可以将移动的这部分石子再移动到奇数级台阶上,后手有必胜策略的情况也如此。对于此题而言,一堆 \(x\) 个石子的 SG 值显然是 \(x\bmod(L+1)\),因此整个游戏的异或值就是所有 \(i\bmod 2=0\) 的 \(a_i\bmod(L+1)\) 的异或和。这个结论放到树上也是成立的,因此对于一次询问 \(v\) 子树的答案而言,如果子树内所有深度与 \(v\) 的深度奇偶性相同的节点的 \(a_i\bmod(L+1)\) 的异或和为 \(0\) 答案就是 GTR,否则答案是 MeiZ。
接下来考虑原问题怎么求解。如果允许离线那我们显然离线建出最终形态的树之后 DFS 序+BIT 即可解决。但是这题不可以离线还要支持插入新点操作,这就使问题变得略有点棘手。LCT 看似可做可我不会。因此考虑 P2137 Gty 的妹子树(怎么又是 Gty/jy)的套路,我们对时间轴分块,每 \(\sqrt{n}\) 次操作重新建树 DFS 一遍,对于已经在重构过的树中的部分的贡献,显然直接 DFS 序+BIT 或者分块即可求出,对于不在重构过的树中的点,我们就暴力开一个 vector,表示上一次重构到这次询问新加入了哪些点,然后每次遍历一遍 vector,在新加入一个点的时候记录一下这个点在重构好的树中哪个点的子树中即可判断该点是否在待查询的点的子树内。如果待查询的点不在重构的树中,那么显然其子树内的点肯定也不在重构好的树中,这样其子树内的点的个数肯定 \(\sqrt{n}\) 级别的。我们就暴力 DFS 找出其子树内的点计算贡献即可。
时间复杂度 \(n\sqrt{n}\)。
讲个笑话,BZOJ 上不重构,复杂度平方竟然过了(
const int MAXN=1e5;
const int BLK=225;
int n,L,a[MAXN+5];link_list<int,MAXN,MAXN*2> g;
int dep[MAXN+5],bgt[MAXN+5],edt[MAXN+5],tim=0,rid[MAXN+5],con[MAXN+5];
bool fin[MAXN+5];int fa[MAXN+5];
void dfs(int x,int f){
rid[bgt[x]=++tim]=x;fin[x]=1;con[x]=x;fa[x]=f;
for(int e=g.hd[x];e;e=g.nxt[e]){
int y=g.val[e];if(y==f) continue;
dep[y]=dep[x]+1;dfs(y,x);
} edt[x]=tim;
}
namespace divblk{
int blk_sz,blk_cnt,L[322],R[322],bel[MAXN+5];
int sum_blk[2][MAXN+5],sum_tot[2][322];
void init(){
blk_sz=316;blk_cnt=317;
for(int i=1;i<=blk_cnt;i++){
L[i]=(i-1)*blk_sz+1;R[i]=min(i*blk_sz,MAXN);
for(int j=L[i];j<=R[i];j++) bel[j]=i;
}
}
void add(int x,int v,int id){
for(int i=x;i<=R[bel[x]];i++) sum_blk[id][i]^=v;
for(int i=bel[x];i<=blk_cnt;i++) sum_tot[id][i]^=v;
}
int query(int x,int id){
if(!x) return 0;
return sum_blk[id][x]^sum_tot[id][bel[x]-1];
}
void clear(){
memset(sum_blk,0,sizeof(sum_blk));
memset(sum_tot,0,sizeof(sum_tot));
}
void rebuild(int n){
clear();
for(int i=1;i<=n;i++){
sum_blk[dep[rid[i]]&1][i]=a[rid[i]];
sum_tot[dep[rid[i]]&1][bel[i]]^=a[rid[i]];
}
for(int _=0;_<2;_++) for(int i=1;i<=blk_cnt;i++){
for(int j=L[i]+1;j<=R[i];j++) sum_blk[_][j]^=sum_blk[_][j-1];
sum_tot[_][i]^=sum_tot[_][i-1];
}
}
}
vector<int> nw;
void rebuild(){
tim=0;memset(fa,0,sizeof(fa));dfs(1,0);
nw.clear();divblk::rebuild(tim);
}
bool in[MAXN+5];
void dfs_fnd(int x,int f){
in[x]=1;
for(int e=g.hd[x];e;e=g.nxt[e]){
int y=g.val[e];if(y==f) continue;
dfs_fnd(y,x);
}
}
void clr(int x,int f){
in[x]=0;
for(int e=g.hd[x];e;e=g.nxt[e]){
int y=g.val[e];if(y==f) continue;
clr(y,x);
}
}
bool vis[MAXN+5];
int main(){
scanf("%d%d",&n,&L);divblk::init();
for(int i=1;i<=n;i++) scanf("%d",&a[i]),a[i]%=(L+1);
for(int i=1,u,v;i<n;i++) scanf("%d%d",&u,&v),g.ins(u,v),g.ins(v,u);
rebuild();
int cnt=0,qu;scanf("%d",&qu);
for(int i=1;i<=qu;i++){
int opt;scanf("%d",&opt);
if(opt==1){
int x;scanf("%d",&x);x^=cnt;
if(fin[x]){
int res=divblk::query(edt[x],~dep[x]&1)^divblk::query(bgt[x]-1,~dep[x]&1);
for(int y:nw){
int z=con[y];
if(bgt[x]<=bgt[z]&&bgt[z]<=edt[x]&&((dep[y]^dep[x])&1)) res^=a[y];
} if(!res) puts("GTY");else puts("MeiZ"),cnt++;
} else {
int res=0;dfs_fnd(x,fa[x]);
for(int y:nw){
if(in[y]&&((dep[y]^dep[x])&1)) res^=a[y];
} if(!res) puts("GTY");else puts("MeiZ"),cnt++;
clr(x,fa[x]);
}
} else if(opt==2){
int x,y;scanf("%d%d",&x,&y);x^=cnt;y^=cnt;y%=(L+1);
if(fin[x]){
divblk::add(bgt[x],a[x],dep[x]&1);a[x]=y;
// printf("!!! %d %d %d\n",bgt[x],x,y);
divblk::add(bgt[x],a[x],dep[x]&1);
} else a[x]=y;
} else {
int u,v,y;scanf("%d%d%d",&u,&v,&y);
u^=cnt;v^=cnt;y^=cnt;y%=(L+1);a[v]=y;fa[v]=u;
assert(!vis[v]);
dep[v]=dep[u]+1;con[v]=con[u];nw.pb(v);
g.ins(u,v);g.ins(v,u);
} if(i%BLK==0) rebuild();
}
return 0;
}
/*
5 3
4 1 2 3 5
1 2
1 3
1 4
1 5
3
1 1
2 4 5
1 0
*/
BZOJ 3729 - Gty的游戏(Staircase 博弈+时间轴分块)的更多相关文章
- BZOJ 3729 GTY的游戏
伪ETT? 貌似就是Splay维护dfn = = 我们首先观察这个博弈 这个博弈直接%(l+1)应该还是很显然的 因为先手怎么操作后手一定能保证操作总数取到(l+1) 于是就变成阶梯Nim了 因为对于 ...
- BZOJ 3729: Gty的游戏 [伪ETT 博弈论]【学习笔记】
题意: 给定一棵有根树,每个节点有一些石子,每次可以将不多于k的石子移动到父节点 修改一个点的石子数,插入一个点,询问某棵子树是否先手必胜 显然是一个阶梯Nim 每次最多取k个,找规律或者观察式子易发 ...
- BZOJ 3729 Gty的游戏 ——Splay
很久很久之前,看到Treap,好深啊 很久之前看到Splay,这数据结构太神了. 之后学习了LCT. 然后看到Top-Tree就更觉得神奇了. 知道我见到了这题, 万物基于Splay 显然需要维护子树 ...
- 【BZOJ 3729】3729: Gty的游戏 (Splay维护dfs序+博弈)
未经博主同意不得转载 3729: Gty的游戏 Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 448 Solved: 150 Description ...
- [BZOJ3729]Gty的游戏
[BZOJ3729]Gty的游戏 试题描述 某一天gty在与他的妹子玩游戏.妹子提出一个游戏,给定一棵有根树,每个节点有一些石子,每次可以将不多于L的石子移动到父节点,询问将某个节点的子树中的石子移动 ...
- bzoj 3991: [SDOI2015]寻宝游戏 虚树 set
目录 题目链接 题解 代码 题目链接 bzoj 3991: [SDOI2015]寻宝游戏 题解 发现每次答案就是把虚树上的路径*2 接在同一关键点上的点的dfs序是相邻的 那么用set动态维护dfs序 ...
- bzoj 3232: 圈地游戏
bzoj 3232: 圈地游戏 01分数规划,就是你要最大化\(\frac{\sum A}{\sum B}\),就二分这个值,\(\frac{\sum A}{\sum B} \geq mid\) \( ...
- [BZOJ 4820] [SDOI2017] 硬币游戏(高斯消元+概率论+字符串hash)
[BZOJ 4820] [SDOI2017] 硬币游戏(高斯消元+概率论+字符串hash) 题面 扔很多次硬币后,用H表示正面朝上,用T表示反面朝上,会得到一个硬币序列.比如HTT表示第一次正面朝上, ...
- [BZOJ 3731] Gty的超级妹子树 (树分块)
[BZOJ 3731] Gty的超级妹子树 (树分块) 题面 给出一棵树(或森林),每个点都有一个值.现在有四种操作 1.查询x子树里>y的值有多少个 2.把点x的值改成y 3.添加一个新节点, ...
随机推荐
- __str__ __repr__区别
当print 实例化对象的时候,可以直接输出__str__ 中的 return结果 在console中 直接输实例对象c 只能输出<__main__.Cycle object at 0x0000 ...
- pycharm运行测试程序提示no tests were found
转载: https://blog.csdn.net/qq_33834328/article/details/75095078
- Hive面试题整理(一)
1.Hive表关联查询,如何解决数据倾斜的问题?(☆☆☆☆☆) 1)倾斜原因:map输出数据按key Hash的分配到reduce中,由于key分布不均匀.业务数据本身的特.建表时考虑不周.等原因 ...
- 三分钟极速体验:Java版人脸检测
欢迎访问我的GitHub https://github.com/zq2599/blog_demos 内容:所有原创文章分类汇总及配套源码,涉及Java.Docker.Kubernetes.DevOPS ...
- SpringCloud 2020.0.4 系列之 Feign
1. 概述 老话说的好:任何问题都有不止一种的解决方法,当前的问题没有解决,只是还没有发现解决方法,而并不是无解. 言归正传,之前我们聊了 SpringCloud 的服务治理组件 Eureka,今天我 ...
- JAVA实现表达式求导运算的分析总结
1第一次作业 1.1题目描述 对形如4*x+x^2+x的多项式求导. 1.2类图 1.3度量分析 在完成第一次作业时,我的写法没有特别的"面向对象".唯一封装起来的是Node,代表 ...
- 设计的MOS管三极管简单开关电路驱动能力不够1
您需要 登录 才可以下载或查看,没有帐号?注册 x . ?& P' U5 r/ ~& `: B 用AOD409设计的开关电路为什么驱动能力不够,请大家帮忙分析一下原因啊.这个电路作用就 ...
- Beam Search快速理解及代码解析
目录 Beam Search快速理解及代码解析(上) Beam Search 贪心搜索 Beam Search Beam Search代码解析 准备初始输入 序列扩展 准备输出 总结 Beam Sea ...
- python的分支结构
python分支结构 if结构 python的 if 选择分支结构的基础语法如下,需要注意的是判断条件后面是半角的分号,它的作用相当于Java中的小括号 if 判断条件 : 代码块 elif 判断条件 ...
- CSS学习笔记:浮动属性
目录 一.浮动流是什么 二.通过代码实例了解浮动特点 1. 搭建测试框架 2. 添加浮动 3. 浮动元素的排布 4. 给行内元素添加浮动效果 5. 子元素浮动后对父元素的影响 5.1 在父元素中添加o ...