[SDOI2013]森林 主席树+启发式合并
这题的想法真的很妙啊。
看到题的第一眼,我先想到树链剖分,并把\(DFS\)序当成一段区间上主席树。但是会发现在询问的时候,可能会非常复杂,因为你需要把路径拆成很多条轻链和重链,它们还不一定连续,很难做(这个做法貌似可以用于子树第\(k\)大问题)。
于是我们换一个思路,让某个点的从它的父亲继承信息,也就是让这个结点对应的主席树维护一条从它到根的链。
那查询该如何是好?假设要查询的结点为\(x,y\),我们利用树上差分的思想让\(x,y,lca(x,y),fa[lca(x,y)]\)对应的四棵主席树一起向下跳就行了。
有连边操作时,就要用到启发式合并了,也就是把小的那棵树合并到大的中,暴力重构一下小树的倍增数组和主席树就行了。
代码:
#include <bits/stdc++.h>
using namespace std;
#define N 100000
#define pb push_back
#define mp make_pair
#define mid ((l+r)>>1)
#define pii pair<int, int>
#define D 17
int n0, n, m, q, T, root[N+5], sz[N+5], d[N+5], me[N+5];
int w0[N+5], w[N+5], as[N+5], f[N+5][20];
int nid, sumv[100*N+5], lson[100*N+5], rson[100*N+5];
int lastans;
vector<int> G[N+5];
void build(int &o, int l, int r)
{
o = ++nid;
if(l == r) return ;
build(lson[o], l, mid), build(rson[o], mid+1, r);
}
int LCA(int x, int y)
{
if(d[x] < d[y]) swap(x, y);
for(int i = D; i >= 0; --i)
if(d[f[x][i]] >= d[y]) x = f[x][i];
if(x == y) return x;
for(int i = D; i >= 0 && f[x][0] != f[y][0]; --i)
if(f[x][i] != f[y][i]) x = f[x][i], y = f[y][i];
return f[x][0];
}
void insert(int o, int &u, int l, int r, int x, int k)
{
u = ++nid;
lson[u] = lson[o], rson[u] = rson[o], sumv[u] = sumv[o]+k;
if(l == r) return ;
if(x <= mid) insert(lson[o], lson[u], l, mid, x, k);
else insert(rson[o], rson[u], mid+1, r, x, k);
}
int query(int x, int y, int k) //x-->y 第k大
{
int lca = LCA(x, y), fa = f[lca][0], l = 1, r = n0;
x = root[x], y = root[y], lca = root[lca], fa = root[fa];
while(l != r)
{
int t = sumv[lson[x]]+sumv[lson[y]]-sumv[lson[fa]]-sumv[lson[lca]];
if(t >= k) r = mid, x = lson[x], y = lson[y], fa = lson[fa], lca = lson[lca];
else k -= t, l = mid+1, x = rson[x], y = rson[y], fa = rson[fa], lca = rson[lca];
}
return as[l];
}
void dfs(int u, int fa, int anc)
{
d[u] = d[fa]+1, sz[u] = 1, f[u][0] = fa, me[u] = anc;
for(int i = 1; i <= D; ++i) f[u][i] = f[f[u][i-1]][i-1];
insert(root[fa], root[u], 1, n0, w[u], 1);
for(int i = 0, v; i < G[u].size(); ++i)
{
v = G[u][i];
if(v == fa) continue;
dfs(v, u, anc);
sz[u] += sz[v];
}
}
void link(int x, int y)
{
G[x].pb(y), G[y].pb(x);
if(sz[me[x]] < sz[me[y]]) dfs(x, y, me[y]);
else dfs(y, x, me[x]);
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("testdata.in", "r", stdin);
freopen("testdata.out", "w", stdout);
#endif
scanf("%d", &T);
scanf("%d%d%d", &n, &m, &q);
for(int i = 1; i <= n; ++i) scanf("%d", &w0[i]), w[i] = w0[i]; //离散化
sort(w0+1, w0+n+1);
n0 = unique(w0+1, w0+n+1)-w0-1;
for(int i = 1, t; i <= n; ++i)
{
t = lower_bound(w0+1, w0+n0+1, w[i])-w0;
as[t] = w[i], w[i] = t;
}
for(int i = 1, x, y; i <= m; ++i)
{
scanf("%d%d", &x, &y);
G[x].pb(y), G[y].pb(x);
}
build(root[0], 1, n0);
for(int i = 1; i <= n; ++i)
if(!d[i]) dfs(i, 0, i);
char c;
for(int i = 1, x, y, z; i <= q; ++i)
{
cin >> c;
scanf("%d%d", &x, &y);
x ^= lastans, y ^= lastans;
if(c == 'Q') scanf("%d", &z), z ^= lastans, printf("%d\n", lastans = query(x, y, z));
else link(x, y);
}
return 0;
}
[SDOI2013]森林 主席树+启发式合并的更多相关文章
- Bzoj 3123: [Sdoi2013]森林(主席树+启发式合并)
3123: [Sdoi2013]森林 Time Limit: 20 Sec Memory Limit: 512 MB Description Input 第一行包含一个正整数testcase,表示当前 ...
- [bzoj3123] [SDOI2013]森林 主席树+启发式合并+LCT
Description Input 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20. 第二行包含三个整数N,M,T,分别表示节点数.初始边数.操作数 ...
- BZOJ 3123: [Sdoi2013]森林 [主席树启发式合并]
3123: [Sdoi2013]森林 题意:一个森林,加边,询问路径上k小值.保证任意时刻是森林 LCT没法搞,树上kth肯定要用树上主席树 加边?启发式合并就好了,小的树dfs重建一下 注意 测试点 ...
- luoguP3302 [SDOI2013]森林 主席树 启发式合并
题目链接 luoguP3302 [SDOI2013]森林 题解 本来这题树上主席树暴力启发式合并就完了 结果把lca写错了... 以后再也不这么写了 复杂度\(O(nlog^2n)\) "f ...
- [BZOJ3123][Sdoi2013]森林 主席树+启发式合并
3123: [Sdoi2013]森林 Time Limit: 20 Sec Memory Limit: 512 MB Description Input 第一行包含一个正整数testcase,表示当 ...
- 【BZOJ 3123】 [Sdoi2013]森林 主席树启发式合并
我们直接按父子关系建主席树,然后记录倍增方便以后求LCA,同时用并查集维护根节点,而且还要记录根节点对应的size,用来对其启发式合并,然后每当我们合并的时候我们都要暴力拆小的一部分重复以上部分,总时 ...
- 【BZOJ-3123】森林 主席树 + 启发式合并
3123: [Sdoi2013]森林 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 2738 Solved: 806[Submit][Status] ...
- P3302 [SDOI2013]森林(主席树+启发式合并)
P3302 [SDOI2013]森林 主席树+启发式合并 (我以前的主席树板子是错的.......坑了我老久TAT) 第k小问题显然是主席树. 我们对每个点维护一棵包含其子树所有节点的主席树 询问(x ...
- BZOJ_3123_[Sdoi2013]森林_主席树+启发式合并
BZOJ_3123_[Sdoi2013]森林_主席树+启发式合并 Description Input 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20 ...
随机推荐
- 在 DotNetCore 3.0 程序中使用通用协议方式启动文件关联应用
问题描述 在传统的基于 .NET Framework 的 WPF 程序中,我们可以使用如下代码段启动相关的默认应用: # 启动默认文本编辑器打开 helloworld.txt Process.Star ...
- Uncaught ReferenceError: jQuery is not defined
页面调试时,明明引入了JQ文件,却一直提示Uncaught ReferenceError: jQuery is not defined错误. 转自:http://blog.csdn.net/baicp ...
- php封装生成随机数函数
随机数函数Random(num,min,max): num:生成的个数 min:最小的数 max:最大的数. <?php //生成随机20个1-80内不重复的随机数 //思路:也没什么思路,就是 ...
- 我的Windows日常——Excel 打开.xls .xlsx 文件格式或文件扩展名无效
就问下各位,这个图,熟不熟?!! 不熟? 好吧当我没问,遇到过的没遇到过的都让我继续写下去吧.... 很多时候,我们新建了一个word文件,但是打开却会弹出这个小窗口,新建的文件出现这个问题,是什么原 ...
- Flink应用案例:How Trackunit leverages Flink to process real-time data from industrial IoT devices
January 22, 2019Use Cases, Apache Flink Lasse Nedergaard Recently there has been significant dis ...
- shell编程企业级实战(2)
Vim配置文件.vimrc vim配置文件 if 条件语句 if是最常见的条件判断语句 例1:如果不存在/backup目录就创建. [root@web-01 /server/tools]# vim 0 ...
- R语言学习——数据合并及绘制密度分布曲线图
setwd("E:/08_cooperation/07_X-lab/06-Crosstalk/Aadapter_primer")# 读取lane01.txt,并对其按列进行相加处理 ...
- marathon传参一
今天试了下marathon传参,新建一个job,增加一个参数,然后用cmd方式,echo出来 定义的json: { "id": "test1", "l ...
- PHP按权重随机
之前业务部门提了一个需求,要求将广告按照ecpm高低进行随机.(即:ecpm高的获取流量的几率大) 如下数组: //要求AD1的概率要求为50%,AD2概率为25% ,AD3的概率为15%,AD4的概 ...
- js实现分段上传文件
使用js实现分段上传文件,本文使用了FileReader对象,可参考:https://developer.mozilla.org/zh-CN/docs/Web/API/FileReader 1)获取文 ...