bzoj 3123 可持久化线段树启发式合并
首先没有连边的操作的时候,我们可以用可持久化线段树来维护这棵树的信息,建立权值可持久化线段树,那么每个点继承父节点的线段树,当询问为x,y的时候我们可以询问rot[x]+rot[y]-rot[lca(x,y)]-rot[lca(x,y)->father]这棵树来得知这个链的信息。
那么对于连边操作,相当于合并两棵树,我们可以将树的节点数小的树全部拆掉连到节点大的树中,这样每个点最多会被操作logn次,每次操作的时间复杂度为logn,所以是mlog^2n的。
反思:对于树的连通性我是用并查集维护的,对于合并操作还需要dfs一次小的树来维护各种信息,但是忘记对x,y点连边了,导致一直RE.(RE是因为某次值不正确,导致下一次^的点超过n)。
各种慢= =。
/**************************************************************
Problem: 3123
User: BLADEVIL
Language: C++
Result: Accepted
Time:16612 ms
Memory:81512 kb
****************************************************************/ //By BLADEVIL
#include <cstdio>
#include <cstring>
#include <algorithm>
#define maxn 100010
#define maxm 200010 using namespace std; struct rec {
int key,num;
rec() {
key=num=;
}
}a[maxn]; struct segment {
int left,right,cnt;
int son[];
segment() {
left=right=cnt=;
memset(son,,sizeof son);
}
}t[maxn<<]; int n,m,l,N,tot;
int pre[maxm<<],last[maxm<<],other[maxm<<];
int ans[maxn],rot[maxn];
int dep[maxn],jump[maxn][],father[maxn],size[maxn],que[maxn]; void connect(int x,int y) {
pre[++l]=last[x];
last[x]=l;
other[l]=y;
} int getfather(int x) {
if (father[x]==x) return x;
return father[x]=getfather(father[x]);
} void dfs(int x,int fa) {
jump[x][]=fa; dep[x]=dep[fa]+;
for (int p=last[x];p;p=pre[p]) {
if (other[p]==fa) continue;
dfs(other[p],x);
}
} bool cmp1(rec x,rec y) {
return x.key<y.key;
} bool cmp2(rec x,rec y) {
return x.num<y.num;
} int lca(int x,int y) {
if (dep[x]>dep[y]) swap(x,y);
int d=dep[y]-dep[x];
for (int i=;i<=;i++) if (d&(<<i)) y=jump[y][i];
if (x==y) return x;
for (int i=;i>=;i--) if (jump[x][i]!=jump[y][i]) x=jump[x][i],y=jump[y][i];
return jump[x][];
} void build(int &x,int l,int r) {
if (!x) x=++tot;
t[x].left=l; t[x].right=r;
if (t[x].left==t[x].right) return ;
int mid=t[x].left+t[x].right>>;
build(t[x].son[],l,mid); build(t[x].son[],mid+,r);
} void insert(int &x,int rot,int key) {
if (!x) x=++tot;
t[x].left=t[rot].left; t[x].right=t[rot].right;
if (t[x].left==t[x].right) {
t[x].cnt=t[rot].cnt+;
return ;
}
int mid=t[x].left+t[x].right>>;
if (key>mid) {
t[x].son[]=t[rot].son[];
insert(t[x].son[],t[rot].son[],key);
} else {
t[x].son[]=t[rot].son[];
insert(t[x].son[],t[rot].son[],key);
}
t[x].cnt=t[rot].cnt+;
} int query(int x,int y,int l1,int l2,int k) {
//printf("%d %d %d ",x,t[x].left,t[x].right);
if (t[x].left==t[x].right) return t[x].left;
int cur=t[t[x].son[]].cnt+t[t[y].son[]].cnt-t[t[l1].son[]].cnt-t[t[l2].son[]].cnt;
//printf("%d\n",cur);
if (k>cur) return query(t[x].son[],t[y].son[],t[l1].son[],t[l2].son[],k-cur); else
return query(t[x].son[],t[y].son[],t[l1].son[],t[l2].son[],k);
} void make(int x,int fa) {
insert(rot[x],rot[fa],a[x].key);
for (int p=last[x];p;p=pre[p]) {
if (other[p]==fa) continue;
make(other[p],x);
}
} void update(int x,int fa,int j) {
jump[x][j]=jump[jump[x][j-]][j-];
for (int p=last[x];p;p=pre[p]) {
if (other[p]==fa) continue;
update(other[p],x,j);
}
} void merge(int x,int y) {
int fx=getfather(x),fy=getfather(y);
//if (x==1) printf("%d %d\n",fx,fy);
if (size[fx]<size[fy]) swap(x,y);
fx=getfather(x),fy=getfather(y);
father[fy]=fx; size[fx]+=size[fy];
dfs(y,x); make(y,x);
for (int i=;i<=;i++) update(y,x,i);
//for (int i=1;i<=n;i++) printf("%d %d\n",i,jump[i][0]);
} int main() {
//freopen("3123.in","r",stdin); freopen("3123.out","w",stdout);
int task; scanf("%d",&task);
scanf("%d%d%d",&n,&m,&task);
for (int i=;i<=n;i++) scanf("%d",&a[a[i].num=i].key); sort(a+,a++n,cmp1);
int cur; ans[]=cur=a[].key;
for (int i=,j=;i<=n;i++)
if (a[i].key==cur) a[i].key=j; else ans[++j]=cur=a[i].key,a[i].key=j,N=j;
sort(a+,a++n,cmp2);
//for (int i=1;i<=n;i++) printf("%d ",ans[i]); printf("\n");
//for (int i=1;i<=n;i++) printf("%d ",a[i].key); printf("\n"); for (int i=;i<=n;i++) father[i]=i,size[i]=;
for (int i=;i<=m;i++) {
int x,y; scanf("%d%d",&x,&y);
connect(x,y); connect(y,x);
int fx=getfather(x),fy=getfather(y);
father[fx]=fy; size[fy]+=size[fx];
}
for (int i=;i<=n;i++) if (!jump[i][]) dfs(i,);
//for (int i=1;i<=n;i++) printf("%d %d\n",i,father[i]);
//for (int i=1;i<=n;i++) if (i==getfather(i)) printf("%d %d\n",i,size[i]); for (int j=;j<=;j++)
for (int i=;i<=n;i++) jump[i][j]=jump[jump[i][j-]][j-]; build(rot[],,N);
for (int i=;i<=n;i++) if (!jump[i][]) make(i,); int ANS=; char Q[];
while (task--) {
scanf("%s",Q);
if (Q[]=='Q') {
int x,y,k; scanf("%d%d%d",&x,&y,&k);
x^=ANS; y^=ANS; k^=ANS;
//printf("|%d %d %d\n",x,y,k);
int rr=lca(x,y); //printf("%d\n",rr);
//for (int i=1;i<=n;i++) printf("%d %d\n",i,jump[i][0]);
printf("%d\n",ANS=ans[query(rot[x],rot[y],rot[rr],rot[jump[rr][]],k)]);
//for (int i=1;i<=n;i++) if (i==getfather(i)) printf("%d %d\n",i,size[i]);
} else {
int x,y; scanf("%d%d",&x,&y);
x^=ANS; y^=ANS; connect(x,y); connect(y,x);
//printf("|%d %d\n",x,y);
merge(x,y);
}
}
return ;
}
bzoj 3123 可持久化线段树启发式合并的更多相关文章
- BZOJ - 3123 森林 (可持久化线段树+启发式合并)
题目链接 先把初始边建成一个森林,每棵树选一个根节点递归建可持久化线段树.当添加新边的时候,把结点数少的树暴力重构,以和它连边的那个点作为父节点继承线段树,并求出倍增数组.树的结点数可以用并查集来维护 ...
- bzoj 3673&3674 可持久化并查集&加强版(可持久化线段树+启发式合并)
CCZ在2015年8月25日也就是初三暑假要结束的时候就已经能切这种题了%%% 学习了另一种启发式合并的方法,按秩合并,也就是按树的深度合并,实际上是和按树的大小一个道理,但是感觉(至少在这题上)更好 ...
- Bzoj 3123: [Sdoi2013]森林(主席树+启发式合并)
3123: [Sdoi2013]森林 Time Limit: 20 Sec Memory Limit: 512 MB Description Input 第一行包含一个正整数testcase,表示当前 ...
- BZOJ 3123: [Sdoi2013]森林 [主席树启发式合并]
3123: [Sdoi2013]森林 题意:一个森林,加边,询问路径上k小值.保证任意时刻是森林 LCT没法搞,树上kth肯定要用树上主席树 加边?启发式合并就好了,小的树dfs重建一下 注意 测试点 ...
- Bzoj 2733: [HNOI2012]永无乡(线段树+启发式合并)
2733: [HNOI2012]永无乡 Time Limit: 10 Sec Memory Limit: 128 MB Description 永无乡包含 n 座岛,编号从 1 到 n,每座岛都有自己 ...
- 【bzoj2653】middle 可持久化线段树区间合并
题目描述 一个长度为n的序列a,设其排过序之后为b,其中位数定义为b[n/2],其中a,b从0开始标号,除法取下整.给你一个长度为n的序列s.回答Q个这样的询问:s的左端点在[a,b]之间,右端点在[ ...
- [CSP-S模拟测试]:模板(ac)(线段树启发式合并)
题目描述 辣鸡$ljh\ NOI$之后就退役了,然后就滚去学文化课了.他每天都被$katarina$大神虐,仗着自己学过一些姿势就给$katarina$大神出了一道题.有一棵$n$个节点的以$1$号节 ...
- [ZJOI2019]语言(树链剖分+动态开点线段树+启发式合并)
首先,对于从每个点出发的路径,答案一定是过这个点的路径所覆盖的点数.然后可以做树上差分,对每个点记录路径产生总贡献,然后做一个树剖维护,对每个点维护一个动态开点线段树.最后再从根节点开始做一遍dfs, ...
- bzoj 3207 可持久化线段树
首先因为固定询问长度,所以我们可以将整个长度为n的数列hash成长度为n-k+1的数列,每次询问的序列也hash成一个数,然后询问这个数是不是在某个区间中出现过,这样我们可以根据初始数列的权值建立可持 ...
随机推荐
- selenium webdriver XPath的定位方法练习 !
html 代码: <html> <body> <div id="div1"> <input name="divl1input& ...
- crontab & php实现多进程思路
<?php $startTime = time(); while(1) { if (time() - $startTime > 600) { exit; } // ... Do SomeT ...
- BZOJ 1565 植物大战僵尸(拓扑排序+最大权闭合子图)
图中的保护关系就类似于最大权闭合子图.即你想杀x,你就一定要杀掉保护x的点,那么把x向保护它的点连边.那么题目就转化成了最大权闭合子图的问题. 但是这个图有点特殊啊... 考虑有环的情况,显然这个环以 ...
- bzoj2253纸箱堆叠(动态规划+cdq分治套树状数组)
Description P 工厂是一个生产纸箱的工厂.纸箱生产线在人工输入三个参数 n p a , 之后,即可自动化生产三边边长为 (a mod P,a^2 mod p,a^3 mod P) (a^4 ...
- Javascript面向对象三大特性(封装性、继承性、多态性)详解及创建对象的各种方法
Javascript基于对象的三大特征和C++,Java面向对象的三大特征一样,都是封装(encapsulation).继承(inheritance )和多态(polymorphism ).只不过实现 ...
- 【刷题】BZOJ 2599 [IOI2011]Race
Description 给一棵树,每条边有权.求一条简单路径,权值和等于K,且边的数量最小.N <= 200000, K <= 1000000 Input 第一行 两个整数 n, k 第二 ...
- Codeforces707Div2
A Small, but very brave, mouse Brain was not accepted to summer school of young villains. He was ups ...
- After ZJOI2017 day2
4.28早上6点左右就起了床,怀着紧张的心情,候到了7:45进考场 看到题,先0.5h看了看题意,yy一下,至少10+20+10. 首先是觉得T3可以搞一搞,先想到SA,很快就X掉了,思索一会儿,感觉 ...
- 百万级运维心得一:Mongodb和Redis数据不能放在同一个服务器
百万级运维经验一:Mongodb和Redis数据不能放在同一个服务器 一开始时,为了省服务器,把Mongodb和Redis放在一个服务器上.网站每到高峰期都特别卡,还经常出现502.找了很久的原因,发 ...
- 洛谷 P1410 子序列(DP)
这题的题解的贪心都是错误的...正解应该是个DP 考虑有哪些有关的条件:两个序列的当前长度, 两个序列的末尾数, 把这些都压进状态显然是会GG的 考虑两个长度加起来那一位的数一定是其中一个序列的末尾, ...