SDOI2013森林
主席树启发式合并,每次连边维护并查集,集合大小,求lca所需信息,合并两个树上的主席树, 重点看代码.
#include <iostream>
#include <algorithm>
#include <cassert>
#include <cstdio>
#include <cstring>
using namespace std;
const int N = 8e4 + 1, M = 2e7 + 1;
int n, m, SIZE, Ht[N], a[N], T;
int head[N], nxt[N<<1], ver[N<<1], tot;
void add(int u, int v)
{ ver[++tot] = v, nxt[tot] = head[u], head[u] = tot; }
int f[N][21], dep[N];
int LCA(int u, int v) {
if (dep[u] < dep[v]) swap(u, v);
for (int i = 20; i >= 0; -- i) {
if (dep[f[u][i]] >= dep[v]) {
u = f[u][i];
}
}
assert(dep[u] == dep[v]);
if (u == v) return u;
for (int i = 20; i >= 0; -- i) {
if (f[u][i] != f[v][i]) {
u = f[u][i];
v = f[v][i];
}
}
assert(f[u][0] == f[v][0]);
return f[u][0];
}
int sum[M], ls[M], rs[M], cnt, rt[N];
void insert(int &p, int l, int r, int val) {
p = ++cnt;
if (l == r) return;
int mid = l + r >> 1;
if (val <= mid) insert(ls[p], l, mid, val);
else insert(rs[p], mid + 1, r, val);
}
void update(int last, int p, int l, int r, int val) {
assert(p);
sum[p] = sum[last] + 1;
if (l ==r) return;
int mid = l + r >> 1;
if (val <= mid) update(ls[last], ls[p], l, mid, val), rs[p] = rs[last];
else update(rs[last], rs[p], mid + 1, r, val), ls[p] = ls[last];
}
int query(int last1, int last2, int p1, int p2, int l, int r, int k) {
if (l == r) return Ht[l];
int s = sum[ls[p1]] - sum[ls[last1]] + sum[ls[p2]] - sum[ls[last2]];
int mid = l + r >> 1;
if (k <= s) return query(ls[last1], ls[last2], ls[p1], ls[p2], l, mid, k);
else return query(rs[last1], rs[last2], rs[p1], rs[p2], mid + 1, r, k - s);
}
bool vis[N];
void DFS(int u, int fa) {
vis[u] = 1;
update(rt[fa], rt[u], 1, SIZE, a[u]);
dep[u] = dep[fa] + 1;
f[u][0] = fa;
for (int i = 1; i <= 20; ++ i)
f[u][i] = f[ f[u][i - 1] ][i - 1];
for (int i = head[u]; i; i = nxt[i]) {
if (ver[i] != fa) DFS(ver[i], u);
}
}
int fa[N], size[N];
int find(int x)
{ return x == fa[x] ? x : fa[x] = find(fa[x]); }
void link(int &u, int &v) {
add(u, v); add(v, u);
int Fu = find(u), Fv = find(v);
if (size[Fu] > size[Fv]) swap(u, v), swap(Fu, Fv);
fa[Fv] = Fu; size[Fu] += size[Fv];
}
int calc(int u, int v, int k) {
int lca = LCA(u, v);
return query(rt[lca], rt[f[lca][0]], rt[u], rt[v], 1, SIZE, k);
}
int main() {
#ifndef ONLINE_JUDGE
freopen("P3302.in", "r", stdin);
freopen("P3302.out", "w", stdout);
#endif
int _;
scanf("%d %d %d %d", &_, &n, &m, &T);
for (int i = 1; i <= n; ++ i) scanf("%d", a + i), Ht[i] = a[i];
sort(Ht + 1, Ht + 1 + n);
SIZE = unique(Ht + 1 ,Ht + 1+ n) - Ht - 1;
for (int i = 1 ;i <= n; ++ i) a[i] = lower_bound(Ht + 1, Ht + 1 + SIZE, a[i]) - Ht;
for (int i = 1; i <= n; ++ i) fa[i] = i, size[i] = 1, insert(rt[i], 1, SIZE, a[i]);
for (int i = 1; i <= m; ++ i) {
int u, v;
scanf("%d %d", &u, &v);
link(u, v);
}
for (int i = 1; i <= n; ++ i)
if (!vis[find(i)]) DFS(find(i), 0);
int ans = 0;
char op[4];
while (T --) {
scanf("%s", op);
if (op[0] == 'Q') {
int x, y, k; scanf("%d%d%d", &x, &y, &k); x ^= ans, y ^= ans, k ^= ans; printf("%d\n", ans = calc(x, y, k));
} else {
int x, y; scanf("%d%d", &x, &y); x ^= ans, y ^= ans; link(x, y); DFS(x, y);
}
}
}
SDOI2013森林的更多相关文章
- BZOJ 3123: [Sdoi2013]森林 [主席树启发式合并]
3123: [Sdoi2013]森林 题意:一个森林,加边,询问路径上k小值.保证任意时刻是森林 LCT没法搞,树上kth肯定要用树上主席树 加边?启发式合并就好了,小的树dfs重建一下 注意 测试点 ...
- luoguP3302 [SDOI2013]森林 主席树 启发式合并
题目链接 luoguP3302 [SDOI2013]森林 题解 本来这题树上主席树暴力启发式合并就完了 结果把lca写错了... 以后再也不这么写了 复杂度\(O(nlog^2n)\) "f ...
- 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] ...
- bzoj 3123: [Sdoi2013]森林(45分暴力)
3123: [Sdoi2013]森林 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 4184 Solved: 1235[Submit][Status ...
- AC日记——[Sdoi2013]森林 bzoj 3123
3123: [Sdoi2013]森林 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 3216 Solved: 944[Submit][Status] ...
随机推荐
- oracle 基础知识(四)常用函数
SQL中的单记录函数 .ASCII 返回与指定的字符对应的十进制数; SQL') zero,ascii(' ') space from dual; A A ZERO SPACE --------- - ...
- 通过ajax给后台提交数据时,radio性别数据的获取
通过ajax向后台异步发送数据,经常我们会遇到个人信息额提交,一般我们采用FormData来装数据.在装性别值得时候,我们会有两个radio框,获取radio值得方法如下: 一般情况下,一个radio ...
- 【HDOJ 1272】小希的迷宫(并查集+无环图)
描述 上次Gardon的迷宫城堡小希玩了很久(见Problem B),现在她也想设计一个迷宫让Gardon来走.但是她设计迷宫的思路不一样,首先她认为所有的通道都应该是双向连通的,就是说如果有一个通道 ...
- VS中R转义字符处理
std::string s1 = R"(Name="Hello World ... ")"; std::string s2 = R"-(Name=&q ...
- Ehcache基于java API实现
上代码: package com.utils.cacheutils; import com.situopenapi.constant.EhcacheConstants; import com.situ ...
- 在Liunx Mint下无法切换到root用户
提示 su: Authentication failure 以ubuntu的mint root用户默认是也是禁止的 需要手动打开才行 a)root启用 执行下面的操作:1.先解除root锁定,为roo ...
- grafana使用json数据源监控数据
功能实现完后有部分数据一直在波动,就产生了想把这个数据波动集成到grafana形成可视化界面的监控,但grafana不支持mongo数据库又懒得去用其他工具转换,特意看了下grafana的databa ...
- MySQL 清除表空间碎片
碎片产生的原因 (1)表的存储会出现碎片化,每当删除了一行内容,该段空间就会变为空白.被留空,而在一段时间内的大量删除操作,会使这种留空的空间变得比存储列表内容所使用的空间更大; (2)当执行插入操作 ...
- 爬虫之爬取斗鱼官网LOL部分主播的状态
一个爬虫小程序 爬取主播的排名及观看人数 import re import requests import request class Spider(): url = 'https://www.dou ...
- STM32进阶之串口环形缓冲区实现(转载)
转载自微信公众号“玩转单片机”,感谢原作者“杰杰”. 队列的概念 在此之前,我们来回顾一下队列的基本概念:队列 (Queue):是一种先进先出(First In First Out ,简称 FIFO) ...