Description

Input

第一行包含一个正整数testcase,表示当前测试数据的测试点编号。保证1≤testcase≤20。 
第二行包含三个整数N,M,T,分别表示节点数、初始边数、操作数。第三行包含N个非负整数表示 N个节点上的权值。 
 接下来 M行,每行包含两个整数x和 y,表示初始的时候,点x和点y 之间有一条无向边, 接下来 T行,每行描述一个操作,格式为“Q x y k”或者“L x y ”,其含义见题目描述部分。

Output

对于每一个第一类操作,输出一个非负整数表示答案。

Sample Input


Q    Q
Q
L
L L
Q Q

Sample Output


HINT

对于第一个操作 Q 8 7 3,此时 lastans=0,所以真实操作为Q 8^0 7^0 3^0,也即Q 8 7 3。点8到点7的路径上一共有5个点,其权值为4 1 1 2 4。这些权值中,第三小的为 2,输出 2,lastans变为2。对于第二个操作 Q 3 5 1 ,此时lastans=2,所以真实操作为Q 3^2 5^2 1^2 ,也即Q 1 7 3。点1到点7的路径上一共有4个点,其权值为 1 1 2 4 。这些权值中,第三小的为2,输出2,lastans变为 2。之后的操作类似。

Solution

时限比较长暗示此题的解法比较暴力

和前面count on a tree的做法一样,先遍历整片森林,初始化倍增数组,把点投到可持久化线段树里去

对于询问操作,一样地,直接递归求解即可

对于连接操作,我们用并查集加size域启发式合并来处理森林的联通状况,方便我们重构树的时候减少重构的点的数量,这样就优化了暴力重构的时间

#include<stdio.h>
#include<string.h>
#define N 80110
#define INF 1000000000
#define mid ((x>>1)+(y>>1)+(x&y&1))
inline void exc(int &x,int &y){
x^=y;y^=x;x^=y;
}
inline int Rin(){
int x=,c=getchar(),f=;
for(;c<||c>;c=getchar())
if(!(c^))f=-;
for(;c>&&c<;c=getchar())
x=(x<<)+(x<<)+c-;
return x*f;
}
int n,m,T,val[N],jump[N][],dep[N],pb[N],top,ans;
struct st{int f,s;}s[N];
inline int pre(int x){
while(s[x].f^x)x=s[x].f;
return x;
}
inline void onion(int x,int y){
x=pre(x),y=pre(y);
s[y].f=x,s[x].s+=s[y].s;
}
struct pt{int v;pt *nxt;}
*fst[N],e[N<<],*tot=e;
inline void link(int x,int y){
*++tot=(pt){y,fst[x]},fst[x]=tot;
*++tot=(pt){x,fst[y]},fst[y]=tot;
}
struct nt{
nt *l,*r;
int s;
}*rt[N],pool[],*C=pool;
inline nt *newnt(nt *_,nt *__,int ___){
C->l=_;C->r=__;C->s=___;
return C++;
}
nt *build(nt *p,int x,int y,int k){
if(!(x^y))return newnt(0x0,0x0,p->s+);
if(k<=mid)return newnt(build(p->l,x,mid,k),p->r,p->s+);
return newnt(p->l,build(p->r,mid+,y,k),p->s+);
}
void dfs(int x){
dep[x]=dep[jump[x][]]+;
rt[x]=build(rt[jump[x][]],,INF,val[x]);
for(pt *j=fst[x];j;j=j->nxt)
if(j->v^jump[x][])
jump[j->v][]=x,
dfs(j->v);
}
void dfs(int x,int f){
pb[++top]=x;
jump[x][]=f;
dep[x]=dep[f]+;
rt[x]=build(rt[f],,INF,val[x]);
for(pt *j=fst[x];j;j=j->nxt)
if(j->v^f)dfs(j->v,x);
}
int lca(int x,int y){
if(dep[x]<dep[y])exc(x,y);
for(int j=;~j;j--)
if(dep[jump[x][j]]>=dep[y])
x=jump[x][j];
if(!(x^y))return x;
for(int j=;~j;j--)
if(jump[x][j]^jump[y][j])
x=jump[x][j],y=jump[y][j];
return jump[x][];
}
int secret(nt *p1,nt *p2,nt *p3,nt *p4,int x,int y,int k){
if(!(x^y))return x;
int c=p1->l->s+p2->l->s-p3->l->s-p4->l->s;
if(k<=c)return secret(p1->l,p2->l,p3->l,p4->l,x,mid,k);
return secret(p1->r,p2->r,p3->r,p4->r,mid+,y,k-c);
}
int feel(int x,int y,int k){
int t=lca(x,y);
return secret(rt[x],rt[y],rt[t],rt[jump[t][]],,INF,k);
}
int main(){
T=Rin(),n=Rin(),m=Rin(),T=Rin();
for(int i=;i<=n;i++)
s[i].f=i,s[i].s=;
for(int i=;i<=n;i++)
val[i]=Rin();
for(int x,y;m;m--)
x=Rin(),y=Rin(),link(x,y),onion(x,y);
rt[]=newnt(C,C,);
for(int i=;i<=n;i++)
if(!jump[i][])
dfs(i);
for(int j=;j<=;j++)
for(int i=;i<=n;i++)
jump[i][j]=jump[jump[i][j-]][j-];
char sign[];
for(int x,y,k;T;T--){
scanf("%s",sign);
x=Rin()^ans,y=Rin()^ans;
if(sign[]=='Q'){
k=Rin()^ans;
printf("%d\n",ans=feel(x,y,k));
}
else{
if(s[pre(x)].s>s[pre(y)].s)
exc(x,y);
top=;
dfs(x,y);
onion(x,y);
for(int j=;j<=;j++)
for(int i=;i<=top;i++)
jump[pb[i]][j]=jump[jump[pb[i]][j-]][j-];
link(x,y);
}
}
return ;
}

[bzoj3123][sdoi2013森林] (树上主席树+lca+并查集启发式合并+暴力重构森林)的更多相关文章

  1. 洛谷P3402 【模板】可持久化并查集 [主席树,并查集]

    题目传送门 可持久化并查集 n个集合 m个操作 操作: 1 a b 合并a,b所在集合 2 k 回到第k次操作之后的状态(查询算作操作) 3 a b 询问a,b是否属于同一集合,是则输出1否则输出0 ...

  2. BZOJ2733[HNOI2012]永无乡——线段树合并+并查集+启发式合并

    题目描述 永无乡包含 n 座岛,编号从 1 到 n,每座岛都有自己的独一无二的重要度,按照重要度可 以将这 n 座岛排名,名次用 1 到 n 来表示.某些岛之间由巨大的桥连接,通过桥可以从一个岛 到达 ...

  3. bzoj3123 [Sdoi2013]森林 树上主席树+启发式合并

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=3123 题解 如果是静态的查询操作,那么就是直接树上主席树的板子. 但是我们现在有了一个连接两棵 ...

  4. SPOJ COT Count on a tree(树上主席树 + LCA 求点第k小)题解

    题意:n个点的树,每个点有权值,问你u~v路径第k小的点的权值是? 思路: 树上主席树就是每个点建一棵权值线段树,具体看JQ博客,LCA用倍增logn求出,具体原理看这里 树上主席树我每个点的存的是点 ...

  5. [BZOJ 3123] [SDOI 2013]森林(可持久化线段树+并查集+启发式合并)

    [BZOJ 3123] [SDOI 2013]森林(可持久化线段树+启发式合并) 题面 给出一个n个节点m条边的森林,每个节点都有一个权值.有两种操作: Q x y k查询点x到点y路径上所有的权值中 ...

  6. 并查集+启发式合并+LCA思想 || 冷战 || BZOJ 4668

    题面:bzoj炸了,以后再补发 题解: 并查集,然后对于每个点记录它与父亲节点联通的时刻 tim ,答案显然是 u 到 v 的路径上最大的 tim 值.启发式合并,把 size 小的子树往大的上并,可 ...

  7. 2017 Multi-University Training Contest - Team 4 phone call(树+lca+并查集)

    题解: (并查集处理往上跳的时候,一定要先让u,v往上跳到并查集的祖先,不然会wa掉) 代码如下: #include <iostream> #include <algorithm&g ...

  8. bzoj 3123 [Sdoi2013]森林(主席树+启发式合并+LCA)

    Description Input 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20. 第二行包含三个整数N,M,T,分别表示节点数.初始边数.操作数 ...

  9. p3302 [SDOI2013]森林(树上主席树+启发式合并)

    对着题目yy了一天加上看了一中午题解,终于搞明白了我太弱了 连边就是合并线段树,把小的集合合并到大的上,可以保证规模至少增加一半,复杂度可以是\(O(logn)\) 合并的时候暴力dfs修改倍增数组和 ...

随机推荐

  1. 完美解决,浏览器下拉显示网址问题 | 完美解决,使用原生 scroll 写下拉刷新

    在 web 开发过程中我们经常遇到,不想让用户下拉看到我的地址,也有时候在 div 中没有惯性滚动,就此也出了 iScroll 这种关于滚动条的框架,但是就为了一个体验去使用一个框架好像又不值得,今天 ...

  2. mybatis笔记1 基本的配置和操作

    mybatis比较轻量,适合开发比较小型的或者业务比较复杂的系统: 相对于hibernate来说可以灵活的写sql,更灵活的处理遇到的业务逻辑: 可以说hibernate是pojo实体对db的orm映 ...

  3. 多线程中的volatile和伪共享

      伪共享 false sharing,顾名思义,“伪共享”就是“其实不是共享”.那什么是“共享”?多CPU同时访问同一块内存区域就是“共享”,就会产生冲突,需要控制协议来协调访问.会引起“共享”的最 ...

  4. java文档注释--javadoc的用法

    1.前言 Java中有三种注释方式.前两种分别是 // 和 /* */,主要用于代码的注释,以此来方便代码的可读性.第三种被称作说明注释或文档注释,它以 /** 开始,以 */结束,文档注释允许你在程 ...

  5. 应用SqlGeometry无法加载sqlserverspatial.dll

    最近需要完成一个API,通过用户上传的经纬度判断用户的所在县市省,数据量相对不是很大所以把相关数据全部扔到了内存里知行,主要用到了SqlGeometry, 代码写完后运行本地没问题,扔到服务器上开始报 ...

  6. Oracle研究专题:Oracle系统安装与配置

    最近开始研究Oracle数据库,盖因公司的系统要么Oracle要么是mysql吧. 作为一个IT工作者,没有碰过Oracle是一件很匪夷所思得事情. 想到过去几年,乃至接触IT行业开始就只有玩过sql ...

  7. GreenDao2.2升级GreenDao3.0的适配之路

    前言.为什么要升级到Greendao3.0? 1. 多人开发 以往的数据库建表建Dao等操作要新开一个module,在统一的地方管理数据库建表,现在可以直接写Entity.多人开发时自己管自己的Ent ...

  8. kindle型号

    这篇文章为 Kindle 新手解决三个问题:怎么辨别Kindle 型号?通过 Kindle 外观判断准确吗?Kindle序列号在那里? 贴吧吧友提问的原话"在咸鱼看到一台,那哥们说也不知道K ...

  9. android:布局、绘制、内存泄露、响应速度、listview和bitmap、线程优化以及一些优化的建议!

    1.布局优化 首先删除布局中无用的控件和层级,其次有选择地使用性能较低的viewgroup,比如布局中既可以使用RelativeLayout和LinearLayout,那我们就采用LinearLayo ...

  10. 了解JavaScript 面向对象基础 & 原型与对象

    面向对象语言中的对象 老是能听到什么基于对象, 面向对象. 什么是对象, 如果有面向对象基础的人可以无视了, 下面举个简单的例子给大家讲讲面向对象中, 对象的定义, 这个是比较通用的, 不过对于JS来 ...