【主席树启发式合并】【P3302】[SDOI2013]森林
Description
给定一个 \(n\) 个节点的森林,有 \(Q\) 次操作,每次要么将森林中某两点联通,保证操作后还是个森林,要么查询两点间权值第 \(k\) 小,保证两点联通。强制在线。
Limitation
\(1~\leq~n,~Q~\leq~80000\)
Solution
考虑有连边还有查询链上第 \(k\) 大,于是要么用 LCT,要么用主席树。
考虑如果用 LCT 的话,并不能快速的维护两点间链的信息(其实感觉在access的时候乱搞一下有希望在多一个 \(\log\) 的代价下维护一颗权值线段树的,但是没有仔细想 ),但是如果使用主席树,在连边的时候可以考虑启发式合并,可以以多一个 \(\log\) 为代价快速合并两个森林。
其实这种合并森林信息的,大概一共就只有 LCT 和启发式合并两种做法吧……
与此类似的在一棵树上合并子树信息的大概只有启发式合并和静态树上链分治两种做法叭……当然不排除有毒瘤题把这个强行转化成子树和父节点连边然后用 LCT 做……启发式合并的例子比如[十二省联考2019]春节十二响,静态树上链分治的例子比如 [CF600E]Lomsat gelral。
于是使用主席树维护每个节点到根的权值线段树即可快速查询链上第 \(k\) 大,在合并森林的时候进行启发式合并。注意到用主席树求链上第 \(k\) 大需要用到两点间 LCA,对于 LCA 的维护可以启发式合并两个森林的倍增数组。合并次数是 \(O(\log n)\) 级别的,每次合并是 \(O(n \log n)\) 的,于是合并总复杂度 \(O(n \log^2 n)\),另外查询复杂度 \(O(q~\log n)\)。所以总的复杂度为 \(O(n~\log^2 n~+~q~\log n)\)。写起来也非常好写,相比于 [HNOI2016]树。
Code
#include <cstdio>
#include <cstring>
#include <algorithm>
#ifdef ONLINE_JUDGE
#define freopen(a, b, c)
#endif
typedef long long int ll;
namespace IPT {
const int L = 1000000;
char buf[L], *front=buf, *end=buf;
char GetChar() {
if (front == end) {
end = buf + fread(front = buf, 1, L, stdin);
if (front == end) return -1;
}
return *(front++);
}
}
template <typename T>
inline void qr(T &x) {
char ch = IPT::GetChar(), lst = ' ';
while ((ch > '9') || (ch < '0')) lst = ch, ch=IPT::GetChar();
while ((ch >= '0') && (ch <= '9')) x = (x << 1) + (x << 3) + (ch ^ 48), ch = IPT::GetChar();
if (lst == '-') x = -x;
}
namespace OPT {
char buf[120];
}
template <typename T>
inline void qw(T x, const char aft, const bool pt) {
if (x < 0) {x = -x, putchar('-');}
int top=0;
do {OPT::buf[++top] = static_cast<char>(x % 10 + '0');} while (x /= 10);
while (top) putchar(OPT::buf[top--]);
if (pt) putchar(aft);
}
const int maxn = 80005;
int lstans, n, m, t;
int ufs[maxn], sz[maxn], MU[maxn], tmp[maxn], anc[18][maxn], depth[maxn];
bool vis[maxn];
struct Tree {
Tree *ls, *rs;
int l, r, v;
~Tree() {
if (this->ls) {
delete this->ls;
delete this->rs;
}
}
Tree() : ls(NULL), rs(NULL), l(0), r(0), v(0) {}
};
Tree *rot[maxn];
struct Edge {
int v;
Edge *nxt;
Edge(const int _v, Edge *h) : v(_v), nxt(h) {}
};
Edge *hd[maxn];
void cont(const int u, const int v) {
hd[u] = new Edge(v, hd[u]);
hd[v] = new Edge(u, hd[v]);
}
void init_hash();
int find(const int x);
int GetLCA(int u, int v);
void dfs(const int u, const int rt);
void build(Tree *u, Tree *pre, const int v);
void buildzero(Tree *u, const int l, const int r);
int query(const Tree *const u, const Tree *const v, const Tree *const x, const Tree *const y, const int k);
int main() {
freopen("1.in", "r", stdin);
qr(lstans); lstans = 0;
qr(n); qr(m); qr(t);
for (int i = 1; i <= n; ++i) {
qr(MU[i]);
}
init_hash();
buildzero(rot[0] = new Tree, 1, n);
for (int i = 1, u, v; i <= m; ++i) {
u = v = 0; qr(u); qr(v);
cont(u, v);
}
for (int i = 1; i <= n; ++i) if (!vis[i]) {
dfs(i, i);
}
int a, b, c;
while (t--) {
char op;
do op = IPT::GetChar(); while ((op != 'Q') && (op != 'L'));
if (op == 'Q') {
a = b = c = 0; qr(a); qr(b); qr(c);
a ^= lstans; b ^= lstans; c ^= lstans;
int k = GetLCA(a, b);
qw(lstans = tmp[query(rot[a], rot[b], rot[k], rot[anc[0][k]], c)], '\n', true);
} else {
a = b = 0; qr(a); qr(b);
a ^= lstans; b ^= lstans;
int fa = find(a), fb = find(b);
if (sz[fa] > sz[fb]) {
std::swap(a, b);
std::swap(fa, fb);
}
sz[fb] += sz[fa];
anc[0][a] = b;
dfs(a, fb);
cont(a, b);
}
}
return 0;
}
void dfs(const int u, const int rt) {
vis[u] = true;
sz[u] = 1; ufs[u] = rt;
depth[u] = depth[anc[0][u]] + 1;
build(rot[u] = new Tree, rot[anc[0][u]], MU[u]);
for (int i = 0; i < 17; ++i) {
anc[i + 1][u] = anc[i][anc[i][u]];
}
for (auto e = hd[u]; e; e = e->nxt) if (e->v != anc[0][u]) {
anc[0][e->v] = u;
dfs(e->v, rt);
sz[u] += sz[e->v];
}
}
void init_hash() {
memcpy(tmp + 1, MU + 1, n << 2);
std::sort(tmp + 1, tmp + 1 + n);
auto ed = std::unique(tmp + 1, tmp + 1 + n);
for (int i = 1; i <= n; ++i) {
MU[i] = std::lower_bound(tmp + 1, ed, MU[i]) - tmp;
}
}
void buildzero(Tree *u, const int l, const int r) {
if ((u->l = l) == (u->r = r)) return;
int mid = (l + r) >> 1;
buildzero(u->ls = new Tree, l, mid); buildzero(u->rs = new Tree, mid + 1, r);
}
void build(Tree *u, Tree *pre, const int v) {
*u = *pre; ++u->v;
if (u->l == u->r) return;
if (u->ls->r >= v) {
build(u->ls = new Tree, pre->ls, v);
} else {
build(u->rs = new Tree, pre->rs, v);
}
}
int GetLCA(int u, int v) {
if (depth[u] > depth[v]) {
std::swap(u, v);
}
int delta = depth[v] - depth[u];
for (int i = 17; delta; --i) if (delta >= (1 << i)) {
delta -= 1 << i;
v = anc[i][v];
}
if (u == v) return u;
for (int i = 17; ~i; --i) if (anc[i][u] != anc[i][v]) {
u = anc[i][u]; v = anc[i][v];
}
return anc[0][v];
}
int query(const Tree *const u, const Tree *const v, const Tree *const x, const Tree *const y, const int k) {
if (u->l == u->r) return u->l;
int lv = u->ls->v + v->ls->v - x->ls->v - y->ls->v;
if (lv >= k) {
return query(u->ls, v->ls, x->ls, y->ls, k);
} else {
return query(u->rs, v->rs, x->rs, y->rs, k - lv);
}
}
inline int find(const int x) {
return ufs[x] == x ? x : ufs[x];
}
Summary
其实这种合并森林信息的,大概一共就只有 LCT 和启发式合并两种做法吧……
与此类似的在一棵树上合并子树信息的大概只有启发式合并和静态树上链分治两种做法叭……
后面想到会再更新的
【主席树启发式合并】【P3302】[SDOI2013]森林的更多相关文章
- P3302 [SDOI2013]森林(主席树+启发式合并)
P3302 [SDOI2013]森林 主席树+启发式合并 (我以前的主席树板子是错的.......坑了我老久TAT) 第k小问题显然是主席树. 我们对每个点维护一棵包含其子树所有节点的主席树 询问(x ...
- BZOJ_3123_[Sdoi2013]森林_主席树+启发式合并
BZOJ_3123_[Sdoi2013]森林_主席树+启发式合并 Description Input 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20 ...
- Bzoj 3123: [Sdoi2013]森林(主席树+启发式合并)
3123: [Sdoi2013]森林 Time Limit: 20 Sec Memory Limit: 512 MB Description Input 第一行包含一个正整数testcase,表示当前 ...
- 【主席树 启发式合并】bzoj3123: [Sdoi2013]森林
小细节磕磕碰碰浪费了半个多小时的时间 Description Input 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20. 第二行包含三个整数N,M ...
- [bzoj3123] [SDOI2013]森林 主席树+启发式合并+LCT
Description Input 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20. 第二行包含三个整数N,M,T,分别表示节点数.初始边数.操作数 ...
- 【bzoj3123】[Sdoi2013]森林 倍增LCA+主席树+启发式合并
题目描述 输入 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20. 第二行包含三个整数N,M,T,分别表示节点数.初始边数.操作数.第三行包含N个非负 ...
- 【BZOJ-3123】森林 主席树 + 启发式合并
3123: [Sdoi2013]森林 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 2738 Solved: 806[Submit][Status] ...
- Bzoj2534:后缀自动机 主席树启发式合并
国际惯例的题面:考虑我们求解出字符串uvu第一个u的右端点为i,第二个u的右端点为j,我们需要满足什么性质?显然j>i+L,因为我们选择的串不能是空串.另外考虑i和j的最长公共前缀(也就是说其p ...
- Bzoj 3673: 可持久化并查集 by zky(主席树+启发式合并)
3673: 可持久化并查集 by zky Time Limit: 5 Sec Memory Limit: 128 MB Description n个集合 m个操作 操作: 1 a b 合并a,b所在集 ...
随机推荐
- 应用Redis分布式锁解决重复通知的问题
研究背景: 这几天被支付宝充值后通知所产生的重复处理问题搞得焦头烂额, 一周连续发生两次重复充钱的杯具, 发事故邮件发到想吐..为了挽回程序员的尊严, 我用了Redis的锁机制. 事故场景: 支付宝下 ...
- tomcat闪退的解决思路
用Tomcat总会遇到启动Tomcat闪退的问题. 什么叫闪退啊,就是闪一下,就退出了控制台. 都闪退了,为啥闪退也不知道呀,又没有错误信息,所以就要先阻止闪退,先看到错误信息,知道启动不起来的原因. ...
- virtual DOM的作用:将DOM的维护工作由系统维护转交给virtual DOM维护
virtual DOM的作用:将DOM的维护工作由系统维护转交给virtual DOM维护 两个方面:对应用端 & 对DOM端(渲染准备的计算) 1.将DOM状态的维护工作由系统维护转交给vi ...
- 【03】Saltstack:远程执行
写在前面的话 远程执行可以说是我们使用 Saltstack 最为基础的目的.所以在这里专门作为单独的一篇来详细的聊聊. 远程执行命令 示例命令: salt '*' cmd.run 'w' 命令分析: ...
- 手写LRU实现
完整基于 Java 的代码参考如下 class DLinkedNode { String key; int value; DLinkedNode pre; DLinkedNode post; } LR ...
- Dos.ORM修改数据遇到的问题
2019年11月6日,今天使用Dos.ORM进行数据的批量修改,出现修改一条数据造成所有数据相应状态改变的情况,代码如下: 按照一步步调试的方式,排查出原因:生成的orm实体类缺少 主键 的标识,该原 ...
- 学Haskell不该误入范畴论
浪费了两个星期去学范畴论,结果没啥用,关键是太抽象了.理解不能. 实际上压根联系也没那么紧密.
- 如何设计提高服务API的安全性(一)基础介绍
场景 现今越来越多公司提供了Sass平台服务,大部分也直接提供API.如快递鸟.微信Api.云服务.如何保证这些服务的安全性是一门重要的课题.如快递跟踪.机票查询等很便捷地影响着我们d的生活,对这些技 ...
- Risc-V简要概括
1.Risc-V硬件平台术语 一个RiscV硬件平台可以包含一个或多个RiscV兼容的核心.其它非RiscV兼容的核心.固定功能的加速器.各种物理存储器结构.I/O设备以及允许这些部件相互连通的互联结 ...
- 精益车间管理如何实现?让APS排程系统来帮忙
精益制造是企业全面的文化改变,它的主要目标是消灭任何形式的浪费.最明显的例子是在生产区域堆积的物料.在制品.等待客户来买的成品.它还可能包括员工不必的移动和不增值的许多流程,目标是在最小的库存,最短的 ...