题目

传送门:QWQ

分析

先敲了个树链剖分,发现无法AC(其实是自己弱,懒得debug、手写栈)

然后去学了学正解

核心挺好理解的,$ query(a) $是$ a $到根的异或和。

答案就是$ lca(x,y) \hat{}  query(x)  \hat{}  query(b) $

接着维护异或和,很显然线段树挺容易搞的。

但我们今天学学树状数组来维护异或和

若将区间$ [l,r] $内的元素全部异或x,相当于在第l位标记x,再在第r+1位标记x,
这样,对于第r位以后的元素,这两个命令互相抵消,查询某个元素的值,只需用树状数组把它之前的命令全部累加起来即可。

强无敌了~~~

代码

lca+dfs序+树状数组:

#include <bits/stdc++.h>
using namespace std;
const int maxn=;
int fa[maxn][],depth[maxn],w[maxn];
int in[maxn],out[maxn],cnt,vis[maxn],n,bit[maxn];
vector<int> G[maxn];
int dfs(int u){
for(int i=;i<;i++){
if(depth[u] < ( << i)) break;
fa[u][i]=fa[fa[u][i-]][i-];
}
in[u]=++cnt;
for(int i=;i<G[u].size();i++){
int v=G[u][i]; if(vis[v]) continue; vis[v]=;
fa[v][]=u; depth[v]=depth[u]+; dfs(v);
} out[u]=cnt;
}
int lca(int x,int y){
if(depth[x]<depth[y]) swap(x,y);
int d=(depth[x]-depth[y]);
for(int i=;i<;i++){
if((<<i) & d) x=fa[x][i];
}
for(int i=;i>=;i--){
if(fa[x][i]!=fa[y][i]){
x=fa[x][i]; y=fa[y][i];
}
}
if(x==y) return x;
else return fa[x][];
}
void add(int x,int v){for(;x<=n;x+=x&-x) bit[x]^=v;}
int query(int x){int ans=;for(;x>;x-=x&-x) ans^=bit[x];return ans;}
int main()
{
int u,v,q;
scanf("%d",&n);
for(int i=;i<=n;i++) scanf("%d",&w[i]);
for(int i=;i<n;i++){
scanf("%d%d",&u,&v);
G[u].push_back(v);G[v].push_back(u);
}
vis[]=;dfs();
for(int i=;i<=n;i++){
add(in[i],w[i]); add(out[i]+,w[i]);
}
scanf("%d",&q);
int a,b; char s[];
while(q--){
scanf("%s%d%d",s,&a,&b);
if(s[]=='Q'){
int ans=w[lca(a,b)]^query(in[a])^query(in[b]);
if(ans!=) puts("Yes");
else puts("No");
// printf("--------------- %d %d %d\n",lca(a,b),query(in[a]),query(in[b]));
}
else{
add(in[a],w[a]);add(out[a]+,w[a]);
w[a]=b;
add(in[a],w[a]);add(out[a]+,w[a]);
}
}
return ;
}
/*
8
1 3 5 2 5 3 1 1
1 5
3 5
2 5
1 4
6 3
7 4
8 3
13
Q 1 2
Q 3 5
Q 1 8
Q 3 5
Q 6 2
C 2 5
C 4 6
C 8 5
Q 1 2
Q 3 5
Q 1 8
Q 3 5
Q 6 2
*/

树链剖分(无法AC):

#include <bits/stdc++.h>
using namespace std;
const int maxn=;
int top[maxn],fa[maxn],son[maxn],dep[maxn],siz[maxn];
int wt[*maxn],w[*maxn],id[*maxn],v[*maxn],cnt,n;
vector<int> G[maxn];
void build(int o,int l,int r){
if(l==r){ v[o]=wt[l]; return; }
int mid=l+r>>;
build(o<<,l,mid); build(o<<|,mid+,r);
v[o]=v[o<<]^v[o<<|];
}
void update(int o,int l,int r,int val,int L){
// printf("--->>> %d %d %d %d %d\n",o,l,r,val,L);
if(l==r){
v[o]=val; return;
}
int mid=l+r>>;
if(L<=mid) update(o<<,l,mid,val,L);
else update(o<<|,mid+,r,val,L);
v[o]=v[o<<]^v[o<<|];
}
int query(int o,int l,int r,int L,int R){
if(l>R||r<L) return ;
if(l>=L&&r<=R){return v[o];}
int mid=l+r>>;
int ans=query(o<<,l,mid,L,R)^query(o<<|,mid+,r,L,R);
return ans;
}
///////////////////////////////////////////
int dfs1(int x,int f,int depth){
dep[x]=depth; fa[x]=f; siz[x]=;
int maxnum=;
for(int i=;i<G[x].size();i++){
int v=G[x][i];
if(v==f) continue;
dfs1(v,x,depth+);
siz[x]+=siz[v];
if(maxnum<siz[v]){
maxnum=siz[v]; son[x]=v;
}
}
}
void dfs2(int x,int topf){
top[x]=topf; id[x]=++cnt; wt[id[x]]=w[x];
if(!son[x]) return;
dfs2(son[x],topf);
for(int i=;i<G[x].size();i++){
int v=G[x][i];
if(v!=son[x]&&v!=fa[x]){
dfs2(v,v);
}
}
}
void debug(){
puts("\ndepth: ");
for(int i=;i<=n;i++) printf("%d ",dep[i]);
puts("\nson: ");
for(int i=;i<=n;i++) printf("%d ",son[i]);
puts("\ntop: ");
for(int i=;i<=n;i++) printf("%d ",top[i]);
}
int Qu(int x,int y){
int ans=;
for(;top[x]!=top[y];){
if(dep[top[x]]<dep[top[y]]) swap(x,y);
ans^=query(,,n,id[top[x]],id[x]);
x=fa[top[x]];
}
if(dep[x]>dep[y]) swap(x,y);
ans^=query(,,n,id[x],id[y]);
return ans;
}
int main()
{
// freopen("1.txt","r",stdin); // int __size__=30<<20;
// char *__p__=(char*)malloc(__size__)+__size__;
// __asm__("movl %0, %%esp\n"::"r"(__p__));
int u,v,q;
scanf("%d",&n);
for(int i=;i<=n;i++) scanf("%d",&w[i]);
for(int i=;i<n;i++){
scanf("%d%d",&u,&v);
G[u].push_back(v);G[v].push_back(u);
}
dfs1(,,); dfs2(,);
build(,,n);
scanf("%d",&q);
// printf("====== %d\n",query(1,1,n,1,1));
int a,b; char s[];
while(q--){
// printf("========= %d\n",q);
scanf("%s%d%d",s,&a,&b);
if(s[]=='Q'){
int ans=Qu(a,b);
if(ans!=) puts("Yes");
else puts("No");
}
else{
update(,,n,b,id[a]);
}
}
// debug();
return ;
}
/*
8
1 3 5 2 5 3 1 1
1 5
3 5
2 5
1 4
6 3
7 4
8 3
13
Q 1 2
Q 3 5
Q 1 8
Q 3 5
Q 6 2
C 2 5
C 4 6
C 8 5
Q 1 2
Q 3 5
Q 1 8
Q 3 5
Q 6 2
*/

【BZOJ】2819: Nim(树链剖分 / lca+dfs序+树状数组)的更多相关文章

  1. Codeforces Round #200 (Div. 1) D Water Tree 树链剖分 or dfs序

    Water Tree 给出一棵树,有三种操作: 1 x:把以x为子树的节点全部置为1 2 x:把x以及他的所有祖先全部置为0 3 x:询问节点x的值 分析: 昨晚看完题,马上想到直接树链剖分,在记录时 ...

  2. 【bzoj3881】[Coci2015]Divljak AC自动机+树链的并+DFS序+树状数组

    题目描述 Alice有n个字符串S_1,S_2...S_n,Bob有一个字符串集合T,一开始集合是空的. 接下来会发生q个操作,操作有两种形式: “1 P”,Bob往自己的集合里添加了一个字符串P. ...

  3. BZOJ 2243:染色(树链剖分+区间合并线段树)

    [SDOI2011]染色Description给定一棵有n个节点的无根树和m个操作,操作有2类:1.将节点a到节点b路径上所有点都染成颜色c:2.询问节点a到节点b路径上的颜色段数量(连续相同颜色被认 ...

  4. BZOJ4012[HNOI2015]开店——树链剖分+可持久化线段树/动态点分治+vector

    题目描述 风见幽香有一个好朋友叫八云紫,她们经常一起看星星看月亮从诗词歌赋谈到 人生哲学.最近她们灵机一动,打算在幻想乡开一家小店来做生意赚点钱.这样的 想法当然非常好啦,但是她们也发现她们面临着一个 ...

  5. 【bzoj1146】[CTSC2008]网络管理Network 倍增LCA+dfs序+树状数组+主席树

    题目描述 M公司是一个非常庞大的跨国公司,在许多国家都设有它的下属分支机构或部门.为了让分布在世界各地的N个部门之间协同工作,公司搭建了一个连接整个公司的通信网络.该网络的结构由N个路由器和N-1条高 ...

  6. BZOJ 3083 遥远的国度(树链剖分+LCA)

    Description 描述zcwwzdjn在追杀十分sb的zhx,而zhx逃入了一个遥远的国度.当zcwwzdjn准备进入遥远的国度继续追杀时,守护神RapiD阻拦了zcwwzdjn的去路,他需要z ...

  7. POJ - 2763 Housewife Wind (树链剖分/ LCA+RMQ+树状数组)

    题意:有一棵树,每条边给定初始权值.一个人从s点出发.支持两种操作:修改一条边的权值:求从当前位置到点u的最短路径. 分析:就是在边可以修改的情况下求树上最短路.如果不带修改的话,用RMQ预处理LCA ...

  8. 【树链剖分】【线段树】bzoj3626 [LNOI2014]LCA

    引用题解: http://blog.csdn.net/popoqqq/article/details/38823457 题目大意: 给出一个n个节点的有根树(编号为0到n-1,根节点为0).一个点的深 ...

  9. cogs 186. [USACO Oct08] 牧场旅行 树链剖分 LCA

    186. [USACO Oct08] 牧场旅行 ★★☆   输入文件:pwalk.in   输出文件:pwalk.out   逐字节对比时间限制:1 s   内存限制:128 MB n个被自然地编号为 ...

随机推荐

  1. JDBC的步骤

    使用jdbc步骤 a.导入数据库厂商提供的驱动程序(导入jar包) b.加载驱动程序 Class.forName("驱动程序类") c.获得连接 Connection conn=D ...

  2. 新浪云使用smarty模板的方法

    在部署到sina app engine(sae)上时出现了问题,因为sae作为云计算平台式无法进行文件读写操作的,所以Smarty中输出的缓存文件就无法实现. 错误信息:“SAE_Fatal_erro ...

  3. Failed to read schema document

    转自:http://eric-yan.iteye.com/blog/1908470 问题描述: web项目web.xml编译错误: schema_reference.4: Failed to read ...

  4. BZOJ2946 Poi2000 公共串 【后缀自动机】

    Description 给出几个由小写字母构成的单词,求它们最长的公共子串的长度. 任务: l 读入单词 l 计算最长公共子串的长度 l 输出结果 Input 文件的第一行是整数 n,1<=n& ...

  5. WPF/UWP 的 Grid 布局竟然有 Bug,还不止一个!了解 Grid 中那些未定义的布局规则

    只要你用 XAML 写代码,我敢打赌你一定用各种方式使(nuè)用(dài)过 Grid.不知你有没有在此过程中看到过 Grid 那些匪夷所思的布局结果呢? 本文将带你来看看 Grid 布局中的 Bu ...

  6. 在iOS中使用ZBar扫描二维码

    最近在做的项目中需要用到二维码扫描功能,之前在Android中使用过ZXing识别二维码,ZXing也有对应的iOS版本,经过了解,ZBar也是一个常用的二维码识别软件,并分别提供了iOS和Andro ...

  7. 剑指offer第二章

    剑指offer第二章 1.二维数组中的查找 在一个二维数组中,每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序.请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含 ...

  8. 重温CLR(十三) 定制特性

    利用定制特性,可宣告式为自己的代码构造添加注解来实现特殊功能.定制特性允许为几乎每一个元数据表记录项定义和应用信息.这种可扩展的元数据信息能在运行时查询,从而动态改变代码的执行方式.使用各种.NET技 ...

  9. Palindrome Degree(hash的思想题)

    个人心得:这题就是要确定是否为回文串,朴素算法会超时,所以想到用哈希,哈希从左到右和从右到左的key值一样就一定是回文串, 那么问题来了,正向还能保证一遍遍历,逆向呢,卡住我了,后面发现网上大神的秦九 ...

  10. LeetCode 549. Binary Tree Longest Consecutive Sequence II

    原题链接在这里:https://leetcode.com/problems/binary-tree-longest-consecutive-sequence-ii/description/ 题目: G ...