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成一个数,然后询问这个数是不是在某个区间中出现过,这样我们可以根据初始数列的权值建立可持 ...
随机推荐
- phaser入手
做phaser小程序,必须先把环境弄好 发现怎么导入都无济于事. 最后决定亲自操刀,在原代码中,引入全局变量.
- JDK1.8最新特性--Lambda表达式(重点)
一个旧版本JDK简单匿名类的用例如下所示: // Java 8之前: JButton show = new JButton("Show"); show.addActionListe ...
- python判断字符串是否包含子字符串
python的string对象没有contains方法,不可以使用string.contains的方法判断是否包含子字符串,但是python有更简单的方法来替换contains函数 python的st ...
- 第113天:Ajax跨域请求解决方法
一.原生JS实现ajax 第一步获得XMLHttpRequest对象 第二步:设置状态监听函数 第三步:open一个连接,true是异步请求 第四部:send一个请求,可以发送一个对象和字符串,不需要 ...
- SDOI2017 解题报告
数字表格 \(T\)次询问,每次给出\(n,m(n,m\le 10^6)\),\(f\)为斐波那契数列,\(f_0=0,f_1=1\),求: \[ \prod _{i=1}^n\prod _{j=1} ...
- poj3107 Godfather 求树的重心
Description Last years Chicago was full of gangster fights and strange murders. The chief of the pol ...
- 【bzoj4305】数列的GCD 组合数学+容斥原理
题目描述 给出一个长度为N的数列{a[n]},1<=a[i]<=M(1<=i<=N). 现在问题是,对于1到M的每个整数d,有多少个不同的数列b[1], b[2], ..., ...
- 病毒侵袭 HDU - 2896(ac自动机 板题)
当太阳的光辉逐渐被月亮遮蔽,世界失去了光明,大地迎来最黑暗的时刻....在这样的时刻,人们却异常兴奋——我们能在有生之年看到500年一遇的世界奇观,那是多么幸福的事儿啊~~ 但网路上总有那么些网站,开 ...
- ms17-010漏洞扫描工具
说明: 1.先利用masscan进行445端口探测 2.利用巡风的脚本对开放445端口的IP进行ms17-010漏洞扫描. 3.使用方法:Python2运行后,按提示输入单个IP或者IP网段. # c ...
- 【Python3的进制扫盲】
一.进制 1.进制简介 进制就是进位制,是人们规定的一种进位方法.计算机底层的数据运算和存储都是二进制数据.计算机语言就是二进制,计算机能直接识别二进制数据,其它数据都不能直接识别. 2.常用进制 对 ...