【BZOJ】2819: Nim(树链剖分 / lca+dfs序+树状数组)
题目
传送门: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序+树状数组)的更多相关文章
- Codeforces Round #200 (Div. 1) D Water Tree 树链剖分 or dfs序
Water Tree 给出一棵树,有三种操作: 1 x:把以x为子树的节点全部置为1 2 x:把x以及他的所有祖先全部置为0 3 x:询问节点x的值 分析: 昨晚看完题,马上想到直接树链剖分,在记录时 ...
- 【bzoj3881】[Coci2015]Divljak AC自动机+树链的并+DFS序+树状数组
题目描述 Alice有n个字符串S_1,S_2...S_n,Bob有一个字符串集合T,一开始集合是空的. 接下来会发生q个操作,操作有两种形式: “1 P”,Bob往自己的集合里添加了一个字符串P. ...
- BZOJ 2243:染色(树链剖分+区间合并线段树)
[SDOI2011]染色Description给定一棵有n个节点的无根树和m个操作,操作有2类:1.将节点a到节点b路径上所有点都染成颜色c:2.询问节点a到节点b路径上的颜色段数量(连续相同颜色被认 ...
- BZOJ4012[HNOI2015]开店——树链剖分+可持久化线段树/动态点分治+vector
题目描述 风见幽香有一个好朋友叫八云紫,她们经常一起看星星看月亮从诗词歌赋谈到 人生哲学.最近她们灵机一动,打算在幻想乡开一家小店来做生意赚点钱.这样的 想法当然非常好啦,但是她们也发现她们面临着一个 ...
- 【bzoj1146】[CTSC2008]网络管理Network 倍增LCA+dfs序+树状数组+主席树
题目描述 M公司是一个非常庞大的跨国公司,在许多国家都设有它的下属分支机构或部门.为了让分布在世界各地的N个部门之间协同工作,公司搭建了一个连接整个公司的通信网络.该网络的结构由N个路由器和N-1条高 ...
- BZOJ 3083 遥远的国度(树链剖分+LCA)
Description 描述zcwwzdjn在追杀十分sb的zhx,而zhx逃入了一个遥远的国度.当zcwwzdjn准备进入遥远的国度继续追杀时,守护神RapiD阻拦了zcwwzdjn的去路,他需要z ...
- POJ - 2763 Housewife Wind (树链剖分/ LCA+RMQ+树状数组)
题意:有一棵树,每条边给定初始权值.一个人从s点出发.支持两种操作:修改一条边的权值:求从当前位置到点u的最短路径. 分析:就是在边可以修改的情况下求树上最短路.如果不带修改的话,用RMQ预处理LCA ...
- 【树链剖分】【线段树】bzoj3626 [LNOI2014]LCA
引用题解: http://blog.csdn.net/popoqqq/article/details/38823457 题目大意: 给出一个n个节点的有根树(编号为0到n-1,根节点为0).一个点的深 ...
- cogs 186. [USACO Oct08] 牧场旅行 树链剖分 LCA
186. [USACO Oct08] 牧场旅行 ★★☆ 输入文件:pwalk.in 输出文件:pwalk.out 逐字节对比时间限制:1 s 内存限制:128 MB n个被自然地编号为 ...
随机推荐
- vue.js 源代码学习笔记 ----- instance proxy
/* not type checking this file because flow doesn't play well with Proxy */ import config from 'core ...
- [转] 用深度学习(CNN RNN Attention)解决大规模文本分类问题 - 综述和实践
转自知乎上看到的一篇很棒的文章:用深度学习(CNN RNN Attention)解决大规模文本分类问题 - 综述和实践 近来在同时做一个应用深度学习解决淘宝商品的类目预测问题的项目,恰好硕士毕业时论文 ...
- 关于Objective-C 2.0 的垃圾收集
Objective-C 2.0最大的增强可能就是垃圾收集了(Garbage Collection).与“垃圾收集”对应的是传统的引用计数(Reference Count)内存管理形式. 使用了垃圾 ...
- 音乐随想——德沃夏克《From The New World》
第一乐章 前奏拖的很长,低音,再低音.突然转向,好像漂泊数月的水手看到了新大陆. 第二乐章 前奏很优美,到双簧管出现的时候宛若紫霞仙子撑船自芦苇荡中飘过. 之后又一段较前奏稍快的旋律,好像看到了梦寐的 ...
- vue中使用less
我们需要下载less对应的依赖包 第一步: 安装less依赖,npm install less less-loader --save 第二步: 修改webpack.config.js文件,配置load ...
- 【排序】希尔排序,C++实现
原创博文,转载请注明出处! 本文代码的github地址 # 基本思路 希尔排序是”直接插入排序“的改进版,也称为“缩小增量排序”.基本原理:先将待排序的数组元素分成多个序列,然后对每个子序 ...
- Git详解之七 自定义Git
以下内容转载自:http://www.open-open.com/lib/view/open1328070404827.html自定义 Git 到目前为止,我阐述了 Git 基本的运作机制和使用方式, ...
- kettle连接oracle出现Error connecting to database: (using class oracle.jdbc.driver.OracleDriver)
jdbc驱动,下载jdbc14.jar文件放入 pdi-ce-5.3.0.0-213\data-integration\libswt\win64里 之后重启kettle即可 jdbc14.jar文 ...
- ruby 的数组操作
转自:http://fujinbing.iteye.com/blog/1126232 1. & [ 1, 1, 3, 5 ] & [ 1, 2, 3 ] # => [1, 3] ...
- My Test about Mat
一.创建Mat >Mat a = cv::Mat(2,2,CV_32S,1); output: [1,1; 1,1] >Mat a = cv::Mat(2,2,CV_32SC3,1); ...