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.添加一个新节点, ...
随机推荐
- 一文看懂JVM内存区域分布与作用
那么我们在开始介绍Java内存区域之前,我们先放一张内存区域的图,方便我们后面介绍的时候可以对照着看. 须知,本文是根据JDK8来介绍的. 程序计数器 首先它是线程私有的,它也称为代码的行号指示器,字 ...
- Alpha-功能规格说明书
项目 内容 这个作业属于哪个课程 2021春季软件工程(罗杰 任健) 这个作业的要求在哪里 团队项目-计划-功能规格说明书 一.引言 1. 项目简介 项目团队:删库跑路对不队 项目名称:题士 项目内容 ...
- [对对子队]会议记录4.21(Scrum Meeting12)
今天已完成的工作 吴昭邦 工作内容:基本实现改变顺序合成 相关issue:实现流水线合成系统的逻辑 相关签入:4.21签入1 梁河览 工作内容:修改设置界面bug 相关签入:4.2 ...
- Spring父子上下文的使用案例
Spring父子上下文的使用案例 一.背景 二.需求 三.实现步骤 1.基础代码编写 2.测试结果 四.小彩蛋 五.完整代码 一.背景 最近在看在使用Spring Cloud的时候发现,当我们通过Fe ...
- 8.18考试总结[NOIP模拟43]
又挂了$80$ 好气哦,但要保持优雅.(草 T1 地衣体 小小的贪心:每次肯定从深度较小的点向深度较大的点转移更优. 模拟一下,把边按链接点的子树最大深度排序,发现实际上只有上一个遍历到的点是对当前考 ...
- 单片机stm32F103单片机晶振不起振的原因分析
这是我在做单片机最小系统板时候碰到的问题,之前虽然也做过相似的板子,可是未曾出现过无源晶振不起振的问题.下面是我在遇到问题后的一些检查,排除问题的过程.本人小菜鸟一个,文章中如有错误和不足,还望各位大 ...
- 常用JAVA API :String 、StringBuilder、StringBuffer的常用方法和区别
摘要 本文将介绍String.StringBuilder类的常用方法. 在java中String类不可变的,创建一个String对象后不能更改它的值.所以如果需要对原字符串进行一些改动操作,就需要用S ...
- TCP/IP简述
一.TCP/IP简述 TCP/IP从字面异议看起来是指TCP和IP两种协议,实际上,它只是利用IP进行通信时必须用到的协议群的统称.具体的来说,IP或ICMP.TCP或UDP.Telnet或FTP.以 ...
- 注意 .NET string.GetHashCode() 用法
需求案例:需要把字符串存入数据库,并且要求数据库中不能有重复的字符串,由此就引出了将字符串hash成特定的hash值,依靠查询hash值是否重复来判断字符串是否重复.这样做的好处在于查询重复字符串的代 ...
- Luogu P2827 [NOIp2016提高组]蚯蚓 | 神奇的队列
题目链接 80分思路: 弄一个优先队列,不停地模拟,切蚯蚓时就将最长的那一条出队,然后一分为二入队,简单模拟即可.还要弄一个标记,表示从开始到当前时间每一条蚯蚓应该加上的长度,操作时就加上,入队时就减 ...