luoguP3302 [SDOI2013]森林
https://www.luogu.org/problemnew/show/P3302
看到查询第 k 小,而且是一颗树,可以联想到在树上的主席树,a 和 b 路径中第 k 小可以通过在 a, b, lca(a, b), fa[lca(a, b)] 四个节点对应的主席树上二分得到
实现主席树是很简单的,连接两个点可以启发式合并,现在最大的问题是连接两个点后还要高效的求出 LCA,一种做法是启发式合并的时候重构倍增数组,也有一种用 LCT 维护 LCA 的方法,相对简单。博主的实现是后者
需要注意的就是找到 fa[lca(a, b)] 的时候,可以模仿 LCT findroot 函数的实现,先 access(lca),然后 splay(lca),找到深度次大的点即可。link 之后 root 会变,进行新操作的时候要 makeroot,才能求出正确的 LCA
总而言之,需要注意的细节还是很多的,具体还是看代码实现吧
#include <bits/stdc++.h>
using namespace std;
typedef unsigned long long ull;
typedef long long ll;
template <typename _T>
inline void read(_T &f) {
f = 0; _T fu = 1; char c = getchar();
while(c < '0' || c > '9') {if(c == '-') fu = -1; c = getchar();}
while(c >= '0' && c <= '9') {f = (f << 3) + (f << 1) + (c & 15); c = getchar();}
f *= fu;
}
const int N = 80000 + 10;
int fa[N], ch[N][2], rev[N], st[N], len;
int n, m, t;
int isroot(int u) {return ch[fa[u]][0] != u && ch[fa[u]][1] != u;}
int get(int u) {return ch[fa[u]][1] == u;}
void pushdown(int u) {
if(rev[u]) {
swap(ch[u][0], ch[u][1]);
rev[ch[u][0]] ^= 1;
rev[ch[u][1]] ^= 1;
rev[u] ^= 1;
}
}
void rotate(int u) {
int old = fa[u], oldd = fa[old], k = get(u);
if(!isroot(old)) ch[oldd][get(old)] = u; fa[u] = oldd;
fa[ch[u][k ^ 1]] = old; ch[old][k] = ch[u][k ^ 1];
fa[old] = u; ch[u][k ^ 1] = old;
}
void splay(int u) {
st[len = 1] = u;
for(int i = u; !isroot(i); i = fa[i]) st[++len] = fa[i];
for(int i = len; i >= 1; i--) pushdown(st[i]);
for(; !isroot(u); rotate(u)) if(!isroot(fa[u])) rotate(get(u) == get(fa[u]) ? fa[u] : u);
}
int access(int u) {
int tmp;
for(int i = 0; u; i = u, u = fa[u]) {
splay(u);
ch[u][1] = i;
tmp = u;
}
return tmp;
}
void makeroot(int u) {
access(u);
splay(u);
rev[u] ^= 1;
}
void link(int x, int y) {
makeroot(x);
fa[x] = y;
}
int LCA(int x, int y) {
access(x);
return access(y);
}
int rt[N], val[N * 600], lc[N * 600], rc[N * 600], a[N], b[N];
int tot = 0;
void build(int &u, int l, int r) {
u = ++tot;
if(l == r) return;
int mid = (l + r) >> 1;
build(lc[u], l, mid);
build(rc[u], mid + 1, r);
}
void ins(int &u, int pre, int l, int r, int x) {
u = ++tot;
val[u] = val[pre] + 1, lc[u] = lc[pre], rc[u] = rc[pre];
if(l == r) return;
int mid = (l + r) >> 1;
if(mid >= x) ins(lc[u], lc[pre], l, mid, x);
else ins(rc[u], rc[pre], mid + 1, r, x);
}
int query(int a, int b, int c, int d, int l, int r, int k) {
if(l == r) return l;
int lsum = val[lc[a]] + val[lc[b]] - val[lc[c]] - val[lc[d]];
int rsum = val[rc[a]] + val[rc[b]] - val[rc[c]] - val[rc[d]];
int mid = (l + r) >> 1;
if(lsum >= k) return query(lc[a], lc[b], lc[c], lc[d], l, mid, k);
else return query(rc[a], rc[b], rc[c], rc[d], mid + 1, r, k - lsum);
}
int blen;
vector <int> g[N];
int siz[N], head[N];
int f[N]; int find(int x) {return f[x] == x ? x : f[x] = find(f[x]);}
inline void addedge(int u, int v) {
g[u].push_back(v);
g[v].push_back(u);
}
void dfs1(int u, int f) {
ins(rt[u], rt[f], 1, blen, a[u]);
for(vector <int> :: iterator it = g[u].begin(); it != g[u].end(); it++) {
int v = *it; if(v == f) continue; dfs1(v, u);
}
}
void Merge(int a, int b) {
int x = find(a), y = find(b);
if(siz[x] < siz[y]) swap(a, b), swap(x, y);
siz[x] += siz[y]; addedge(a, b);
link(a, b); f[y] = x;
dfs1(b, a);
}
int lastans = 0;
int main() {
cin >> n;
cin >> n >> m >> t;
for(int i = 1; i <= n; i++) {
scanf("%d", &a[i]);
b[i] = a[i];
f[i] = i; siz[i] = 1;
}
sort(b + 1, b + n + 1);
blen = unique(b + 1, b + n + 1) - b - 1;
for(int i = 1; i <= n; i++) a[i] = lower_bound(b + 1, b + blen + 1, a[i]) - b;
build(rt[0], 1, blen);
for(int i = 1; i <= n; i++) {
ins(rt[i], rt[0], 1, blen, a[i]);
}
for(int i = 1; i <= m; i++) {
int A, B;
scanf("%d %d", &A, &B);
Merge(A, B);
}
for(int i = 1; i <= t; i++) {
char c = getchar();
while(c != 'Q' && c != 'L') c = getchar();
if(c == 'L') {
int A, B;
scanf("%d %d", &A, &B);
A ^= lastans; B ^= lastans;
Merge(A, B);
} else {
int A, B, K;
scanf("%d %d %d", &A, &B, &K);
A ^= lastans; B ^= lastans; K ^= lastans;
makeroot(find(A));
int lca = LCA(A, B), father;
if(find(A) == lca) father = 0;
else {
access(lca); splay(lca);
father = ch[lca][0];
while(ch[father][1]) {
father = ch[father][1];
pushdown(father);
}
}
printf("%d\n", lastans = b[query(rt[A], rt[B], rt[lca], rt[father], 1, blen, K)]);
}
}
return 0;
}
luoguP3302 [SDOI2013]森林的更多相关文章
- luoguP3302 [SDOI2013]森林 主席树 启发式合并
题目链接 luoguP3302 [SDOI2013]森林 题解 本来这题树上主席树暴力启发式合并就完了 结果把lca写错了... 以后再也不这么写了 复杂度\(O(nlog^2n)\) "f ...
- [luoguP3302] [SDOI2013]森林(主席树 + 启发式合并 + lca)
传送门 显然树上第k大直接主席树 如果连边的话,我们重构小的那一棵,连到另一棵上. 说起来简单,调了我一晚上. 总的来说3个错误: 1.离散化写错位置写到了后面 2."="写成了& ...
- BZOJ 3123: [Sdoi2013]森林 [主席树启发式合并]
3123: [Sdoi2013]森林 题意:一个森林,加边,询问路径上k小值.保证任意时刻是森林 LCT没法搞,树上kth肯定要用树上主席树 加边?启发式合并就好了,小的树dfs重建一下 注意 测试点 ...
- P3302 [SDOI2013]森林(主席树+启发式合并)
P3302 [SDOI2013]森林 主席树+启发式合并 (我以前的主席树板子是错的.......坑了我老久TAT) 第k小问题显然是主席树. 我们对每个点维护一棵包含其子树所有节点的主席树 询问(x ...
- [BZOJ3123][Sdoi2013]森林 主席树+启发式合并
3123: [Sdoi2013]森林 Time Limit: 20 Sec Memory Limit: 512 MB Description Input 第一行包含一个正整数testcase,表示当 ...
- BZOJ3123: [Sdoi2013]森林(启发式合并&主席树)
3123: [Sdoi2013]森林 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 4813 Solved: 1420[Submit][Status ...
- 【BZOJ3123】[Sdoi2013]森林 主席树+倍增LCA+启发式合并
[BZOJ3123][Sdoi2013]森林 Description Input 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20. 第二行包含三个整 ...
- 洛谷 P3302 [SDOI2013]森林 解题报告
P3302 [SDOI2013]森林 题目描述 小\(Z\)有一片森林,含有\(N\)个节点,每个节点上都有一个非负整数作为权值.初始的时候,森林中有\(M\)条边. 小Z希望执行\(T\)个操作,操 ...
- 3123: [Sdoi2013]森林
3123: [Sdoi2013]森林 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 3336 Solved: 978[Submit][Status] ...
随机推荐
- java成神之——正则表达式基本使用
正则表达式 常用匹配规则 基本使用 标记符的使用 部分正则标记 正则表达式在字符串方法中的使用 结语 正则表达式 常用匹配规则 [abc] abc其中一个 [^abc] abc之外的一个 [a-z] ...
- phpStudy启动失败时的解决方法 提示缺vc9运行库
问题描述: 问题产生原因分析: php5.3.5.4和apache都是用vc9编译,电脑必须安装vc9运行库才能运行. php5.5.5.6是vc11编译,如用php5.5.5.6必须安装vc11运行 ...
- Centos安装php高版本
安装 1.检查当前是否有安装php rpm -qa|grep php 如果有安装PHP,那么请先删除这些安装包: yum remove php* 2.安装php源 Centos 5 安装php源: ...
- [C#] 等待启动的进程执行完毕
有能有时候我们启动了一个进程,必须等到此进程执行完毕,或是,一段时间, 关闭进程后再继续往下走. Example sample1 等待应用程序执行完毕 //等待应用程序执行完毕 private voi ...
- itemize,enumerate,description 用法【LaTeX 使用】
itemize和enumerate还有description 是LaTeX里列举的三种样式,分别讲一些使用技巧.itemize(意为分条目):\begin{itemize}\item[*] a\ite ...
- ZooKeeper集群搭建过程
ZooKeeper集群搭建过程 提纲 1.ZooKeeper简介 2.ZooKeeper的下载和安装 3.部署3个节点的ZK伪分布式集群 3.1.解压ZooKeeper安装包 3.2.为每个节点建立d ...
- SQLServer中的事物与锁
了解事务和锁 事务:保持逻辑数据一致性与可恢复性,必不可少的利器. 锁:多用户访问同一数据库资源时,对访问的先后次序权限管理的一种机制,没有他事务或许将会一塌糊涂,不能保证数据的安全正确读写. 死锁: ...
- elasticsearch2.x插件之一:marvel(配置)
Marvel是Elastic公司推出的商业监控方案,是用来监控Elasticsearch集群,历史状态的有力工具,便于性能优化以及故障诊断.监控主要分为六个层面,分别是集群层.节点层.索引层.分片层. ...
- EF添加和修改
(1)//添加操作 public bool addDate() { try { //声明上下文 a_context = new AEntities(); //声明数据模型实体 //执行代码时候会先验证 ...
- 模板模式和Comparable类
模板模式中,父类规定好了一些算法的流程,并且空出一些步骤(方法)留给子类填充 Java的数组类中静态方法sort()就是一个模板,它空出了一个compareTo的方法,留给子类填充,用来规定什么是大于 ...