->题目链接

题解:

#include<queue>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#define MAXN 160001
#define D 15
using namespace std;
char ch[];
int t,x,y,z;
int n,m,qq,tot,cut,ans,father[MAXN],dis[MAXN];
int head[MAXN],size[MAXN],deep[MAXN],root[MAXN],s[MAXN],fa[MAXN][D+];
struct edge {
int v,next;
} e[MAXN*];
struct data {
int lc,rc,size;
} tree[MAXN*];
struct node {
int x,o;
} a[MAXN];
void add(int u,int v) {
e[++cut].v=v;
e[cut].next=head[u];
head[u]=cut;
}
bool cmp(const node &x,const node &y) {
if(x.x==y.x) return x.o<y.o;
return x.x<y.x;
}
int find(int x) {
return x!=father[x]?father[x]=find(father[x]):x;
}
void change(int &k,int last,int l,int r,int x) {
k=++tot;
tree[k].lc=tree[last].lc,tree[k].rc=tree[last].rc;
tree[k].size=tree[last].size+;
if(l==r) return ;
int mid=(l+r)>>;
if(x<=mid) change(tree[k].lc,tree[last].lc,l,mid,x);
else change(tree[k].rc,tree[last].rc,mid+,r,x);
return ;
}
int query(int L,int R,int lc,int falc,int l,int r,int k) {
if(l==r) return l;
int sum=tree[tree[R].lc].size+tree[tree[L].lc].size-
tree[tree[lc].lc].size-tree[tree[falc].lc].size;
int mid=l+r>>;
if(sum>=k)
return query(tree[L].lc,tree[R].lc,tree[lc].lc,tree[falc].lc,l,mid,k);
else return
query(tree[L].rc,tree[R].rc,tree[lc].rc,tree[falc].rc,mid+,r,k-sum);
}
void dfs1(int u,int f) {
size[u]=;
fa[u][]=f;
int p=lower_bound(s+,s+n+,a[u].x)-s;
change(root[u],root[f],,n,p);
for(int i=head[u]; i; i=e[i].next) {
int v=e[i].v;
if(v!=f) deep[v]=deep[u]+,dfs1(v,u),size[u]+=size[v];
}
return ;
}
void get_father() {
for(int j=; j<=D; j++)
for(int i=; i<=n; i++)
fa[i][j]=fa[fa[i][j-]][j-];
return ;
}
int get_same(int u,int x) {
for(int i=; i<=D; i++)
if(x&(<<i)) u=fa[u][i];
return u;
}
int lca(int x,int y) {
if(deep[x]<deep[y]) swap(x,y);
x=get_same(x,deep[x]-deep[y]);
if(x==y) return x;
for(int i=D; i>=; i--)
if(fa[x][i]!=fa[y][i]) x=fa[x][i],y=fa[y][i];
return fa[x][];
}
void slovequery(int x,int y,int k) {
int lc=lca(x,y);
ans=query(root[x],root[y],root[lc],root[fa[lc][]],,n,k);
ans=s[ans];
printf("%d\n",ans);
return ;
}
void slove(int u,int f) {
fa[u][]=f;
deep[u]=deep[f]+;
int p=lower_bound(s+,s+n+,a[u].x)-s;
change(root[u],root[f],,n,p);
for(int i=; i<=D; i++) fa[u][i]=fa[fa[u][i-]][i-];
for(int i=head[u]; i; i=e[i].next) {
int v=e[i].v;
if(v!=f) slove(v,u);
}
return ;
}
void union_(int x,int y) {
int l1=find(x),l2=find(y);
if(size[l1]>size[l2]) swap(l1,l2),swap(x,y);
add(x,y),add(y,x);
father[l1]=l2,size[l1]+=size[l2];
slove(x,y);
return ;
}
int main() {
cin>>t>>n;
cin>>m>>qq;
for(int i=; i<=n; i++)
father[i]=i,cin>>a[i].x,a[i].o=i,s[i]=a[i].x;
sort(s+,s+n+);
for(int i=; i<=m; i++) {
cin>>x>>y,add(x,y),add(y,x);
int l1=find(x),l2=find(y);
father[l2]=l1;
}
for(int i=; i<=n; i++) if(!fa[i][]) dfs1(i,);
get_father();
while(qq--) {
scanf("%s",ch);
if(ch[]=='Q') {
cin>>x>>y>>z;
x^=ans,y^=ans,z^=ans;
slovequery(x,y,z);
} else {
cin>>x>>y;
x^=ans,y^=ans;
union_(x,y);
}
}
return ;
}

一世安宁

洛谷 P3302 [SDOI2013]森林的更多相关文章

  1. 洛谷 P3302 [SDOI2013]森林 解题报告

    P3302 [SDOI2013]森林 题目描述 小\(Z\)有一片森林,含有\(N\)个节点,每个节点上都有一个非负整数作为权值.初始的时候,森林中有\(M\)条边. 小Z希望执行\(T\)个操作,操 ...

  2. 洛谷 P3302 [SDOI2013]森林 Lebal:主席树 + 启发式合并 + LCA

    题目描述 小Z有一片森林,含有N个节点,每个节点上都有一个非负整数作为权值.初始的时候,森林中有M条边. 小Z希望执行T个操作,操作有两类: Q x y k查询点x到点y路径上所有的权值中,第k小的权 ...

  3. [bzoj3123][洛谷P3302] [SDOI2013]森林(树上主席树+启发式合并)

    传送门 突然发现好像没有那么难……https://blog.csdn.net/stone41123/article/details/78167288 首先有两个操作,一个查询,一个连接 查询的话,直接 ...

  4. P3302 [SDOI2013]森林(主席树+启发式合并)

    P3302 [SDOI2013]森林 主席树+启发式合并 (我以前的主席树板子是错的.......坑了我老久TAT) 第k小问题显然是主席树. 我们对每个点维护一棵包含其子树所有节点的主席树 询问(x ...

  5. [洛谷P3304] [SDOI2013]直径

    洛谷题目链接:[SDOI2013]直径 题目描述 小Q最近学习了一些图论知识.根据课本,有如下定义.树:无回路且连通的无向图,每条边都有正整数的权值来表示其长度.如果一棵树有N个节点,可以证明其有且仅 ...

  6. Bzoj3197/洛谷3296 [SDOI2013]刺客信条assassin(树的重心+树Hash+树形DP+KM)

    题面 Bzoj 洛谷 题解 (除了代码均摘自喻队的博客,可是他退役了) 首先固定一棵树,枚举另一棵树,显然另一棵树只有与这棵树同构才有可能产生贡献 如果固定的树以重心为根,那么另一棵树最多就只有重心为 ...

  7. 洛谷$P3302$ 森林 $[SDOI2013]$ 主席树

    正解:主席树 解题报告: 传送门! 口胡一时爽代码火葬场 这题想法不难,,,但显然的是代码应该还挺难打的 但反正我也不放代码,就写下题解趴$QwQ$ 第一问就是个$Count\ on\ a\ tree ...

  8. 洛谷P3302 森林

    题意:给定森林,可以把两棵树连起来或者询问链上第k大. 解:启发式合并. 我一开始想到了启发式合并但是发现这样做之后一棵子树就不是一段连续的区间了,那就不能子树xxx了,很迷惘. 后来看了题解发现本来 ...

  9. 洛谷 P3299 [SDOI2013]保护出题人 解题报告

    P3299 [SDOI2013]保护出题人 题目描述 出题人铭铭认为给SDOI2012出题太可怕了,因为总要被骂,于是他又给SDOI2013出题了. 参加SDOI2012的小朋友们释放出大量的僵尸,企 ...

随机推荐

  1. [Android] 仿照 Last App Switcher 写的小程序

      在Android众多工具类app中,Last App Switcher绝对算是一个让人用过就不会卸载的工具.LAS这个应用,它的功能很简单,就是通过一个浮动按钮实现在两个应用之间一键切换,但是非常 ...

  2. 蓝魔i7s刷机

    ,电脑管家,豌豆荚之类的PC工具 5. 安装MFT6.0.43.exe, 注意:MFT6.0.43需要对应的ISOC和USB驱动,不可使用其它版本 6. 将CUSTOM_CONFIG..INI文件复制 ...

  3. 如何在 Azure 中标记 Windows 虚拟机

    本文介绍在 Azure 中通过 Azure 资源管理器标记 Windows 虚拟机的不同方式.标记是用户定义的键/值对,可直接放置在资源或资源组中.针对每个资源和资源组,Azure 当前支持最多 15 ...

  4. Sql server 账号被锁住:"the account is currently locked out. The system administrator can unlock it."的解决办法(转载)

    今天遇到的问题比较有意思.首先是很久没有打开测试数据库了,今天打开,使用service程序测试的时候出现下面的错误提示:Message: System.Data.SqlClient.SqlExcept ...

  5. .Net WebRequest异步请求与WebClient异步请求

    很多情况下一般会使用同步方式发出请求,直到响应后再做后续的逻辑处理等,但有时候后续的逻辑处理不依赖于请求的结果或者是可以挂起等到响应后再处理,又或者是为了解决UI“假死”的现象,这时可以使用异步请求 ...

  6. 转:C#综合揭秘——细说多线程(上)

    原文地址:http://www.cnblogs.com/leslies2/archive/2012/02/07/2310495.html 引言 本文主要从线程的基础用法,CLR线程池当中工作者线程与I ...

  7. ZooKeeper 的读写操作 & 选举机制

    0. 说明 记录 ZooKeeper 的读写操作和选举机制 1. ZooKeeper 的读写操作 读操作:所有 ZooKeeper 节点都可以提供读请求(包括 follower 和 leader ) ...

  8. windows10下安装source insight 4.0(破解版)

    1.从官网下载source insight4.0版本(不用下载,在后面已经把所有需要的文件都准备好了); 2.安装source insightt4.0; 3.使用下载好的sourceinsight4. ...

  9. [C++] stack和queue的常用函数

    参考资料: STL 在 OI 中的应用 stack stack 后入先出(LIFO)栈 头文件: #include<stack> 定义: stack<int> s; 函数: 函 ...

  10. 1、JUC--volatile 关键字-内存可见性

    Java JUC简介 在 Java 5.0 提供了 java.util.concurrent (简称JUC )包,在此包中增加了在并发编程中很常用的实用工具类,用于定义类似于线程的自定义子系统,包括线 ...