BZOJ:2819 NIM(树链剖分||DFS序 &&NIM博弈)
著名游戏设计师vfleaking,最近迷上了Nim。普通的Nim游戏为:两个人进行游戏,N堆石子,每回合可以取其中某一堆的任意多个,可以取完,但不可以不取。谁不能取谁输。这个游戏是有必胜策略的。于是vfleaking决定写一个玩Nim游戏的平台来坑玩家。
为了设计漂亮一点的初始局面,vfleaking用以下方式来找灵感:拿出很多石子,把它们聚成一堆一堆的,对每一堆编号1,2,3,4,...n,在堆与堆间连边,没有自环与重边,从任意堆到任意堆都只有唯一一条路径可到达。然后他不停地进行如下操作:
1.随机选两个堆v,u,询问若在v到u间的路径上的石子堆中玩Nim游戏,是否有必胜策略,如果有,vfleaking将会考虑将这些石子堆作为初始局面之一,用来坑玩家。
2.把堆v中的石子数变为k。
由于vfleaking太懒了,他懒得自己动手了。请写个程序帮帮他吧。
题意:给定一棵树,每个节点上有个数值,现在Q个操作:
一种操作是给定u,v,然后需要回答u到v的最短路径上的这些数的Nim博弈结果。
一种操作是给定x,val,然后需要把x节点上的数值改为val。
思路:1,树剖,就不多说了。
2,dfs序可以得到u到根root的节点信息,所以sum(u,v)=sum(u,LCA)^sum(v,son[LCA])=sum(1,u)^sum(1,fa[v])。
具体的得到根到节点信息,可以参考这道题:https://blog.csdn.net/clove_unique/article/details/70213935
虽然第二种简单很多,这里树剖练习,先不管它:
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
const int maxn=;
int a[maxn],sum[maxn<<],top[maxn],sz[maxn],son[maxn];
int Laxt[maxn],Next[maxn<<],To[maxn<<],cnt,N,Q;
int fa[maxn],dep[maxn],times,tid[maxn],Rank[maxn];
void add(int u,int v)
{
Next[++cnt]=Laxt[u];
Laxt[u]=cnt;
To[cnt]=v;
}
void dfs1(int u,int pre)
{
sz[u]=; fa[u]=pre; dep[u]=dep[pre]+;
for(int i=Laxt[u];i;i=Next[i]){
int v=To[i];
if(v==pre) continue;
dfs1(v,u);
sz[u]+=sz[v];
if(sz[v]>sz[son[u]]) son[u]=v;
}
}
void dfs2(int u,int Top)
{
tid[u]=++times; Rank[times]=u; top[u]=Top;
if(son[u]) dfs2(son[u],Top);
for(int i=Laxt[u];i;i=Next[i]){
int v=To[i];
if(v==fa[u]||v==son[u]) continue;
dfs2(v,v);
}
}
void update(int Now)
{
sum[Now]=sum[Now<<]^sum[Now<<|];
}
void build(int Now,int L,int R)
{
if(L==R){
sum[Now]=a[Rank[L]];
return ;
}
int Mid=(L+R)>>;
build(Now<<,L,Mid);
build(Now<<|,Mid+,R);
update(Now);
}
int getsum(int Now,int L,int R,int l,int r)
{
if(l<=L&&r>=R) return sum[Now];
int Mid=(L+R)>>,res=;
if(l<=Mid) res^=getsum(Now<<,L,Mid,l,r);
if(r>Mid) res^=getsum(Now<<|,Mid+,R,l,r);
return res;
}
int query(int u,int v)
{
int res=;
while(top[u]!=top[v]){
if(dep[top[u]]<dep[top[v]]) swap(u,v);
res^=getsum(,,N,tid[top[u]],tid[u]);
u=fa[top[u]];
}
if(dep[u]>dep[v]) swap(u,v);
res^=getsum(,,N,tid[u],tid[v]);
return res;
}
void change(int Now,int L,int R,int pos,int val)
{
if(L==R){
sum[Now]=val; return ;
}
int Mid=(L+R)>>;
if(pos<=Mid) change(Now<<,L,Mid,pos,val);
else change(Now<<|,Mid+,R,pos,val);
update(Now);
}
int main()
{
int u,v,i; char opt[];
scanf("%d",&N);
for(i=;i<=N;i++) scanf("%d",&a[i]);
for(i=;i<N;i++){
scanf("%d%d",&u,&v);
add(u,v); add(v,u);
}
dfs1(,); dfs2(,); build(,,N);
scanf("%d",&Q);
while(Q--){
scanf("%s%d%d",opt,&u,&v);
if(opt[]=='Q'){
if(query(u,v)) printf("Yes\n");
else puts("No");
}
else change(,,N,tid[u],v);
}
return ;
}
BZOJ:2819 NIM(树链剖分||DFS序 &&NIM博弈)的更多相关文章
- BZOJ 3083: 遥远的国度(树链剖分+DFS序)
		可以很显而易见的看出,修改就是树链剖分,而询问就是在dfs出的线段树里查询最小值,但由于这道题会修改根节点,所以在查询的时候需判断x是否为root的祖先,如果不是就直接做,是的话应该查询从1-st[y ... 
- BZOJ 3083: 遥远的国度 [树链剖分 DFS序 LCA]
		3083: 遥远的国度 Time Limit: 10 Sec Memory Limit: 1280 MBSubmit: 3127 Solved: 795[Submit][Status][Discu ... 
- [BZOJ - 2819] Nim 【树链剖分 / DFS序】
		题目链接: BZOJ - 2819 题目分析 我们知道,单纯的 Nim 的必胜状态是,各堆石子的数量异或和不为 0 .那么这道题其实就是要求求出树上的两点之间的路径的异或和.要求支持单点修改. 方法一 ... 
- 树链剖分&dfs序
		树上问题 很多处理区间的问题(像是RMQ,区间修改).可以用线段树,树状数组,ST表这些数据结构来维护.但是如果将这些问题挪到了树上,就不能直接用这些数据结构来处理了.这时就用到了dfs序和树链剖分. ... 
- BZOJ_4034 [HAOI2015]树上操作 【树链剖分dfs序+线段树】
		一 题目 [HAOI2015]树上操作 二 分析 树链剖分的题,这里主要用到了$dfs$序,这题比较简单的就是不用求$lca$. 1.和树链剖分一样,先用邻接链表建双向图. 2.跑两遍$dfs$,其实 ... 
- 树链剖分||dfs序 各种题
		1.[bzoj4034][HAOI2015]T2 有一棵点数为 N 的树,以点 1 为根,且树点有边权.然后有 M 个 操作,分为三种: 操作 1 :把某个节点 x 的点权增加 a . 操作 2 :把 ... 
- BZOJ 4196: [Noi2015]软件包管理器 [树链剖分 DFS序]
		4196: [Noi2015]软件包管理器 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 1352 Solved: 780[Submit][Stat ... 
- BZOJ 2286 树链剖分+DFS序+虚树+树形DP
		第一次学习虚树,就是把无关的点去掉.S里维护一条链即可. #include <iostream> #include <cstring> #include <cstdio& ... 
- BZOJ - 4196 软件包管理器 (树链剖分+dfs序+线段树)
		题目链接 设白色结点为未安装的软件,黑色结点为已安装的软件,则: 安装软件i:输出结点i到根的路径上的白色结点的数量,并把结点i到根的路径染成黑色.复杂度$O(nlog^2n)$ 卸载软件i:输出结点 ... 
随机推荐
- 【Kubernetes】Kubernetes删除namespace后持续terminating状态
			删除isti和foo的配置文件之后,namespace持续terminating状态,此时也无法再创建istio-system的namespace namespace "istio-syst ... 
- POJ 1159 字符串匹配问题
			题目大意: 问至少添加几个字符才能保证这个字符串是个回文串 一开始想也想不到字符串匹配上,因为是找回文串,我们可以把已给字符串逆向得到一个新的字符串,然后比较两者得到最大匹配长度,最后总长度减去最大匹 ... 
- 次最短路径 POJ 3255 Roadblocks
			http://poj.org/problem?id=3255 这道题还是有点难度 要对最短路径的算法非常的了解 明晰 那么做适当的修改 就可以 关键之处 次短的路径: 设u 到 v的边权重为cost ... 
- BZOJ2059: [Usaco2010 Nov]Buying Feed 购买饲料
			数轴上n<=500个站可以买东西,每个站位置Xi,库存Fi,价格Ci,运东西价格是当前运载重量的平方乘距离,求买K<=10000个东西到达点E的最小代价. f[i,j]--到第i站不买第i ... 
- Linux中命令选项及参数简介
			登录Linux后,我们就可以在#或$符后面去输入命令,有的时候命令后面还会跟着“选项”(英文options)或“参数”(英文arguments).即Linux中命令格式为: command [opti ... 
- Python基础教程笔记——第4章:字典
			字典 字典是Python唯一内建的数学映射类型,字典中的值没有特殊的顺序,键可以是数字,字符串,甚至是元组 字典的创建: 字典由键值对构成,字典中键是唯一的,而值不唯一.>>> a_ ... 
- msp430项目编程06
			msp430中项目---设计扫描键盘 1.扫描键盘工作原理 2.电路原理说明 3.代码(显示部分) 4.代码(键盘驱动) 5.项目总结 msp430项目编程 msp430入门学习 
- ThinkPHP5 的入门学习
			与Tp3.2相比,有一下的不同: (1)目录名称的改变: tp3.2的目录命名首字母皆为大写,例如:Application.Public.Controller.Model.View.ThinkPHP. ... 
- Python高级进阶(一)Python框架之Django入门
			传说中的Django Django由来 Django是一个开放源代码的Web应用框架,由Python写成.采用了MVC的框架模式,即模型M,视图V和控制器C.它最初是被开发来用于管理劳伦斯出版集团旗下 ... 
- [Bzoj3233][Ahoi2013]找硬币[基础DP]
			3233: [Ahoi2013]找硬币 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 924 Solved: 482[Submit][Status][ ... 
