Description

给你一片森林, 支持两个操作: 查询$x$到$y$的$K$大值,  连接两棵树中的两个点

Solution

对每个节点$x$动态开权值线段树, 表示从$x$到根节点路径上权值出现的次数。

查询时差分即可: $sum[x]+sum[y]-sum[lca]-sum[f[lca]]$

连边时需要启发式合并,将节点数小的接到节点数大的上去, 再遍历小的树, 并更新权值

我刚开始以为testcase是数据组数, TLE我好久,,

Code

 #include<cstdio>
#include<cstring>
#include<algorithm>
#define rd read()
using namespace std; const int N = 1e5; int lson[N * ], rson[N * ], sum[N * ], root[N];
int n, m, T, a[N], b[N], f[N][], head[N], tot, dep[N];
int father[N], num[N], cnt, nd_num;
int lastans, Case; struct edge {
int nxt, to;
}e[N << ]; int read() {
int X = , p = ; char c = getchar();
for(; c > '' || c < ''; c = getchar()) if(c == '-') p = -;
for(; c >= '' && c <= ''; c = getchar()) X = X * + c - '';
return X * p;
} void add(int u, int v) {
e[++tot].to = v;
e[tot].nxt = head[u];
head[u] = tot;
} int find_anc(int x) {
return father[x] == x ? x : father[x] = find_anc(father[x]);
} int find_lca(int x, int y) {
if(dep[x] < dep[y]) swap(x, y);
for(int i = ; ~i; --i) if(dep[f[x][i]] >= dep[y])
x = f[x][i];
if(x == y) return x;
for(int i = ; ~i; --i) if(f[x][i] != f[y][i])
x = f[x][i], y = f[y][i];
return f[x][];
} int fd(int x) {
return lower_bound(b + , b + + cnt, x) - b;
} void ins(int &o, int now, int l, int r, int pos) {
o = ++nd_num;
sum[o] = sum[now] + ;
lson[o] = lson[now];
rson[o] = rson[now];
if(l == r) return;
int mid = (l + r) >> ;
if(pos <= mid) ins(lson[o], lson[now], l, mid, pos);
else ins(rson[o], rson[now], mid + , r, pos);
} int query(int x, int y, int lca, int flca, int l, int r, int k) {
if(l == r) return l;
int mid = (l + r) >> , tmp;
if((tmp = sum[lson[x]] + sum[lson[y]] - sum[lson[lca]] - sum[lson[flca]]) >= k) return query(lson[x], lson[y], lson[lca], lson[flca], l, mid, k);
else return query(rson[x], rson[y], rson[lca], rson[flca], mid + , r, k - tmp);
} void dfs(int u) {
dep[u] = dep[f[u][]] + ;
for(int i = ; i <= ; ++i)
f[u][i] = f[f[u][i - ]][i - ];
ins(root[u], root[f[u][]], , cnt, fd(a[u]));
for(int i = head[u]; i; i = e[i].nxt) {
int nt = e[i].to;
if(nt == f[u][]) continue;
f[nt][] = u;
dfs(nt);
}
} int work() {
lastans = ; tot = ;
nd_num = ;
/* memset(root, 0, sizeof(root));
memset(lson, 0, sizeof(lson));
memset(rson, 0, sizeof(rson));
memset(head, 0, sizeof(head));
memset(dep, 0, sizeof(dep));*/
n = rd; m = rd; T = rd;
for(int i = ; i <= n; ++i) b[i] = a[i] = rd;
sort(b + , b + + n);
cnt = unique(b + , b + + n) - b - ;
for(int i = ; i <= n; ++i) father[i] = i, num[i] = ;
for(int i = ; i <= m; ++i) {
int u = rd, v = rd;
int x = find_anc(u), y = find_anc(v);
father[y] = x;
num[x] += num[y];
add(u, v); add(v, u);
}
for(int i = ; i <= n; ++i) if(!dep[i]) dfs(i);
for(int i = ; i <= T; ++i) {
char c = getchar();
while(c != 'Q' && c != 'L') c = getchar();
int u = rd ^ lastans, v = rd ^ lastans;
if(c == 'Q') {
int lca = find_lca(u, v), flca = f[lca][], k = rd ^ lastans;
lastans = query(root[u], root[v], root[lca], root[flca], , cnt, k);
if(lastans > cnt || lastans < ) return printf("F**k,WA\n"), ;
lastans = b[lastans];
printf("%d\n", lastans);
}
else {
int x = find_anc(u), y = find_anc(v);
if(num[x] < num[y]) {
swap(x, y); swap(u, v);
}
father[y] = x;
num[x] += num[y];
f[v][] = u;
add(v, u); add(u, v);
dfs(v);
}
}
return ;
} int main()
{
Case = rd;
work();
}

BZOJ 3123 [SDOI2013] 森林 - 启发式合并 主席树的更多相关文章

  1. 【BZOJ3123】[SDOI2013] 森林(启发式合并主席树)

    点此看题面 大致题意: 给你一片森林,有两种操作:询问两点之间的第\(k\)小点权和在两棵树之间连一条边. 前置技能:树上主席树 做这道题目,我们首先要会树上主席树. 关于树上主席树,这有一道很好的例 ...

  2. bzoj 3674: 可持久化并查集加强版 (启发式合并+主席树)

    Description Description:自从zkysb出了可持久化并查集后……hzwer:乱写能AC,暴力踩标程KuribohG:我不路径压缩就过了!ndsf:暴力就可以轻松虐!zky:…… ...

  3. BZOJ 2733 [HNOI2012]永无乡 - 启发式合并主席树

    Description 1: 查询一个集合内的K大值 2: 合并两个集合 Solution 启发式合并主席树板子 Code #include<cstdio> #include<cst ...

  4. BZOJ 3123: [Sdoi2013]森林 [主席树启发式合并]

    3123: [Sdoi2013]森林 题意:一个森林,加边,询问路径上k小值.保证任意时刻是森林 LCT没法搞,树上kth肯定要用树上主席树 加边?启发式合并就好了,小的树dfs重建一下 注意 测试点 ...

  5. Bzoj 3123: [Sdoi2013]森林(主席树+启发式合并)

    3123: [Sdoi2013]森林 Time Limit: 20 Sec Memory Limit: 512 MB Description Input 第一行包含一个正整数testcase,表示当前 ...

  6. BZOJ3123: [Sdoi2013]森林(启发式合并&主席树)

    3123: [Sdoi2013]森林 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 4813  Solved: 1420[Submit][Status ...

  7. bzoj 3123: [Sdoi2013]森林(45分暴力)

    3123: [Sdoi2013]森林 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 4184  Solved: 1235[Submit][Status ...

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

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

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

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

随机推荐

  1. ADO.Net 综合练习题

    题目: 第一部分: 新建一个数据库:ADO测试,包含下面两个数据表,使用代码创建,并保留创建的代码文本. 专业表Subject: 专业编号(SubjectCode):nvarchar类型,不能为空,主 ...

  2. vue 解决页面闪烁问题

    watch: { //监听表格数据的变化[使用 watch+nextTick 可以完成页面数据监听的 不会出现闪烁] tableData: { //深度监听,可监听到对象.数组的变化 handler( ...

  3. JMeter学习(十四)JMeter函数学习(转载)

    转载自 http://www.cnblogs.com/yangxia-test JMeter函数是一些能够转化在测试树中取样器或者其他配置元件的域的特殊值.一个函数的调用就像这样:${_functio ...

  4. PHP使用UTF8编码读取ACCESS的乱码问题解决方案(转)

    PHP使用UTF8编码读取ACCESS的乱码问题解决方案 http://it.xwstudy.com/readnews.php?id=627 来源:本站编辑 发布日期:2013-05-27 已有 17 ...

  5. try cache

    try{ $did = DB::insert('vmi_sales_orders',array_keys($value))->values($value)->execute('newerp ...

  6. 已知一个函数rand7()能够生成1-7的随机数,请给出一个函数rand10(),该函数能够生成1-10的随机数。

    题目: 已知一个函数rand7()能够生成1-7的随机数,请给出一个函数,该函数能够生成1-10的随机数. 思路: 假如已知一个函数能够生成1-49的随机数,那么如何以此生成1-10的随机数呢? 解法 ...

  7. Floyd算法简介

    参考:https://blog.csdn.net/qq_35644234/article/details/60875818 一.Floyd算法的介绍    1.算法的特点:    弗洛伊德算法是解决任 ...

  8. 第三章 列表(a)接口与实现

  9. mycat配置实现mysql读写分离

    需要先把mysql的主从复制配置好,然后才可以开始mycat的配置 m ysql主从复制配置:https://www.cnblogs.com/renjianjun/p/9093062.html myc ...

  10. HDU-1459.非常可乐(BFS )

    这道题TLE了很多次,原来一直以为将数字化为最简可以让运算更快,但是去了简化之后才发现,真正耗时的就是化简....还和队友学到了用状态少直接数组模拟刚就能过... 本题大意:给出可乐的体积v1,给出两 ...