[BZOJ4730][清华集训2016][UOJ266] Alice和Bob又在玩游戏
题意:俩智障又在玩游戏。规则如下:
给定n个点,m条无向边(m<=n-1),保证无环,对于每一个联通块,编号最小的为它们的根(也就是形成了一片这样的森林),每次可以选择一个点,将其本身与其祖先全部删除,不能操作者输。判断先手胜负。
题解:比较神的一道题。
我们现在要解决的问题是怎么求解一棵子树的SG值,首先把根删掉的情况考虑,这很好办,直接把子树的sg异或起来就好,关键是如果删除点在子树里怎么办。
这里用到了一个巧妙的东西,trie。怎么会用这个呢?因为删除子树里的节点就相当于是子树里这种对应的情况再异或上外边子树的sg。但是我们不可能用一般的方法来存一棵子树里所有的sg。这个时候trie应运而生。我们处理子树之后,把它合并上来,就能得到当前节点的所有拓展局面的sg了。这里注意,合并子树前要先在子树上打一个tag(因为它是要异或上外面所有子树sg的)。
算法很清晰了,dfs下去,合并上来。这里的trie还要打tag。所有细节就这么多。
#include<bits/stdc++.h>
using namespace std;
#define LL long long
#define N 100005 inline LL read(){
LL x=,f=; char a=getchar();
while(a<'' || a>'') {if(a=='-') f=-; a=getchar();}
while(a>='' && a<='') x=x*+a-'',a=getchar();
return x*f;
} int n,m,T,bin[],head[N],cnt,id,tag[N*],rt[N],ls[N*],rs[N*],sg[N],sz[N*];
bool vis[N]; struct edges{
int to,next;
}e[*N]; inline void insert(){
int u=read(),v=read();
e[++cnt]=(edges){v,head[u]};head[u]=cnt;
e[++cnt]=(edges){u,head[v]};head[v]=cnt;
} inline void init(){
bin[]=; for(int i=;i<=;i++) bin[i]=bin[i-]<<;
} inline void pushdown(int k,int level){
if(!tag[k]) return;
if(bin[level-]&tag[k]) swap(ls[k],rs[k]);
tag[ls[k]]^=tag[k]; tag[rs[k]]^=tag[k];
tag[k]=;
} inline void reset(){
for(int i=;i<=n;i++) head[i]=sg[i]=rt[i]=,vis[i]=;
for(int i=;i<=id;i++) tag[i]=ls[i]=rs[i]=sz[i]=;
cnt=; id=;
} void ins(int& k,int x,int level){ // 0 is on the left
k=++id; sz[k]=;
if(!level) return;
if(x&bin[level-]) ins(rs[k],x,level-);
else ins(ls[k],x,level-);
} int merge(int x,int y,int level){
if(!x || !y) return x|y;
pushdown(x,level); pushdown(y,level);
ls[x]=merge(ls[x],ls[y],level-); rs[x]=merge(rs[x],rs[y],level-);
sz[x]=sz[ls[x]]+sz[rs[x]]+(level?:);
return x;
} void dfs(int x,int fa){
vis[x]=; int t=;
for(int i=head[x];i;i=e[i].next){
if(fa==e[i].to) continue;
dfs(e[i].to,x); t^=sg[e[i].to];
}
ins(rt[x],t,);
for(int i=head[x];i;i=e[i].next){
if(fa==e[i].to) continue;
tag[rt[e[i].to]]^=t^sg[e[i].to];
rt[x]=merge(rt[x],rt[e[i].to],);
}
for(int now=rt[x],i=;i;i--){ // i is the i th digit int binary system
pushdown(now,i);
if(sz[ls[now]]<bin[i-]) now=ls[now];
else sg[x]|=bin[i-],now=rs[now];
}
} inline void solve(){
n=read(); m=read(); int ans=;
for(int i=;i<=m;i++) insert();
for(int i=;i<=n;i++) if(!vis[i]) dfs(i,),ans^=sg[i];
puts(ans?"Alice":"Bob");
} int main(){
init(); T=read();
while(T--){
solve();
reset();
}
return ;
}
[BZOJ4730][清华集训2016][UOJ266] Alice和Bob又在玩游戏的更多相关文章
- UOJ #266 【清华集训2016】 Alice和Bob又在玩游戏
题目链接:Alice和Bob又在玩游戏 这道题就是一个很显然的公平游戏. 首先\(O(n^2)\)的算法非常好写.暴力枚举每个后继计算\(mex\)即可.注意计算后继的时候可以直接从父亲转移过来,没必 ...
- UOJ#266. 【清华集训2016】Alice和Bob又在玩游戏 博弈,DSU on Tree,Trie
原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ266.html 题解 首先我们可以直接暴力 $O(n^2)$ 用 sg 函数来算答案. 对于一个树就是枚举 ...
- uoj#266. 【清华集训2016】Alice和Bob又在玩游戏(博弈论)
传送门 完了我连sg函数是个啥都快忘了 设\(sg[u]\)为以\(u\)为根节点的子树的\(sg\)函数值,\(rem[u]\)表示\(u\)到根节点的路径删掉之后剩下的游戏的异或值 根节点\(u\ ...
- 【清华集训2016】Alice和Bob又在玩游戏
不难的题目.因为SG性质,所以只需要对一棵树求出. 然后如果发现从上往下DP不太行,所以从下往上DP. 考虑一个点对子树的合并,考虑下一个删的点在哪一个子树,那么剩下的状态实际上就是把一个子树所有能达 ...
- UOJ 266 - 【清华集训2016】Alice和Bob又在玩游戏(SG 定理+01-trie)
题面传送门 神仙题. 首先注意到此题的游戏是一个 ICG,故考虑使用 SG 定理解决这个题,显然我们只需对每个连通块计算一遍其 SG 值异或起来检验是否非零即可.注意到我们每删除一个点到根节点的路径后 ...
- [UOJ266]Alice和Bob又在玩游戏
[UOJ266]Alice和Bob又在玩游戏 Tags:题解 作业部落 评论地址 TAG:博弈 题意 不同于树的删边游戏,删掉一个点删去的是到根的路径 题解 这题只和计算\(SG\)有关,博弈的有关内 ...
- uoj266[清华集训2016]Alice和Bob又在玩游戏(SG函数)
uoj266[清华集训2016]Alice和Bob又在玩游戏(SG函数) uoj 题解时间 考虑如何求出每棵树(子树)的 $ SG $ . 众所周知一个状态的 $ SG $ 是其后继的 $ mex $ ...
- bzoj4730: Alice和Bob又在玩游戏
Description Alice和Bob在玩游戏.有n个节点,m条边(0<=m<=n-1),构成若干棵有根树,每棵树的根节点是该连通块内编号最 小的点.Alice和Bob轮流操作,每回合 ...
- 【bzoj4730】 Alice和Bob又在玩游戏
http://www.lydsy.com/JudgeOnline/problem.php?id=4730 (题目链接) 题意 给出一个森林,两个人轮流操作,每次把一个节点以及它的祖先全部抹去,无节点可 ...
随机推荐
- PHP输出语句大杂烩
一 echo echo() 实际上不是一个函数,是php语句,因此您无需对其使用括号.不过,如果您希望向 echo() 传递一个以上的参数,那么使用括号会发生解析错误.而且echo是返回void的,并 ...
- Linux 进程(二):进程关系及其守护进程
进程关系 进程组 进程组是一个或多个进程的集合.通常,它们是在同一作业中结合起来的,同一进程组中的各进程接收来自同一终端的各种信号,每个进程组有一个唯一的进程组ID.每个进程组有一个组长进程,该组长进 ...
- django restframework 的日常使用
本文讨论 django restframework 的日常使用,满足常用 api 编写的需求,比如 List, Detail, Update, Put, Patch 等等.探讨 django rest ...
- 面向对象、接口编程的重要性 python 为什么引入接口interface
面向对象编程的实践 有个产品,其有10个子产品,现在要统计每日消费数据其中8个子产品的消费入账金额算法相同,2个不同; 拓展性差的糟糕的代码 def ConsumptionSum(product): ...
- 谷歌浏览器input中的text 和 button 水平对齐的问题
方法一 text 的vertical-align :top; 方法二 button的vertical-align: middle;
- 设计一个算法,採用BFS方式输出图G中从顶点u到v的最短路径(不带权的无向连通图G採用邻接表存储)
思想:图G是不带权的无向连通图.一条边的长度计为1,因此,求带顶点u和顶点v的最短的路径即求顶点u和顶点v的边数最少的顶点序列.利用广度优先遍历算法,从u出发进行广度遍历,类似于从顶点u出发一层一层地 ...
- Java性能监控之javassist探索
https://www.cnblogs.com/orionhp/p/6362615.html ApplicationContext实现的默认行为就是在启动时将所有singleton bean提前进行实 ...
- Android studio 相关下载
Android studio http://www.androiddevtools.cn/ Oracle的VirtulBox https://www.virtualbox.org/wiki/Down ...
- 剑指offer 面试55题
面试55题: 题目:二叉树的深度 题:输入一棵二叉树,求该树的深度.从根结点到叶结点依次经过的结点(含根.叶结点)形成树的一条路径,最长路径的长度为树的深度. 解题思路: ①如果一棵树只有一个节点,它 ...
- Windows.old
如果通过执行自定义安装来安装 Windows 7,而没有在安装过程中格式化分区,则以前版本的 Windows中使用的文件存储在 Windows.old文件夹中.此文件夹中文件的类型取决于您的电脑.使用 ...