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.添加一个新节点, ...
随机推荐
- Egg.js学习与实战系列 · Post请求`csrf token`问题
在使用axios请求egg.js封装的post接口时出现missing csrf token 或 invalid csrf token.踩过坑的新手估计不在少数,本篇记录一下解决方法. 问题原因 引用 ...
- BUAA软件工程:软件案例分析
BUAA软件工程:软件案例分析 Author:17373015 乔玺华 项目 内容 这个作业属于哪个课程 2020计算机学院软件工程(罗杰 任健) 这个作业的要求在哪里 软件案例分析博客作业 我在这个 ...
- Spring Cloud Gateway GatewayFilter的使用
Spring Cloud Gateway GatewayFilter的使用 一.GatewayFilter的作用 二.Spring Cloud Gateway内置的 GatewayFilter 1.A ...
- elasticsearch的索引操作
1.创建索引(test_index) curl -XPUT "http://192.168.99.1:9200/test_index" 2.创建索引,指定分片和副本的数量 curl ...
- 最短路spaf及dijkstra模板
spaf的双端队列优化: #include<bits/stdc++.h> #define ll long long const ll maxn=210000; using namespac ...
- Python 模块 itertools
python 2.6 引入了itertools模块,使得排列组合的实现非常简单: import itertools 有序排列:e.g., 4个数内选2个排列: >>> print l ...
- 数组模拟双链表,你get到了吗?
数组模拟双链表 通过前面的学习我们知道单链表是单个指针指向操作,那么通过类比我们可以把指针设定为两个,并且让它们分别指向前后数据,这就是"双向链表".使用这种链表,不仅可以从前往后 ...
- etcd安装常用操作
etcd安装 etcd 是基于 Raft 的分布式 key-value 存储系统,由 CoreOS 开发,常用于服务发现.共享配置以及并发控制(如 leader 选举.分布式锁等).kubernete ...
- RF运行之后控制信息日志显示乱码(解决方法)
RIDE运行自动化测试案例时,控制台信息显示乱码,如下所示: 解决方法:进入到python的安装路径下:D:\python37\Lib\site-packages\robotide\contrib\t ...
- spring security 之自定义表单登录源码跟踪
上一节我们跟踪了security的默认登录页的源码,可以参考这里:https://www.cnblogs.com/process-h/p/15522267.html 这节我们来看看如何自定义单表认 ...