luoguP3302 [SDOI2013]森林 主席树 启发式合并
题目链接
题解
本来这题树上主席树暴力启发式合并就完了
结果把lca写错了...
以后再也不这么写了
复杂度\(O(nlog^2n)\)
"for(int i = 0;dad[x][i];++ i) dad[x][i+1] = dad[dad[x][i]][i]"
De了两个多小时....QAQ
代码
#include<cstdio>
#include<cstring>
#include<algorithm>
inline int read() {
int x = 0,f = 1;
char c = getchar();
while(c < '0' || c > '9') {if(c == '-')f = -1;c = getchar(); }
while(c <= '9' && c >= '0')x = x * 10 + c - '0',c = getchar();
return x * f;
}
const int maxn = 100007;
struct NNnode {
int v,next;
} edge[maxn << 2];
int num = 0,head[maxn];
inline void add_edge(int u,int v) {
edge[++ num].v = v; edge[num].next = head[u]; head[u] = num;
}
int root[maxn],val[maxn],n,m,t,MX;
int tot = 0;
int ls[40000000],rs[40000000],cnt[40000000];
void insert(int pre,int &now,int l,int r,int key) {
cnt[++ tot] = cnt[now]; now = tot;
++ cnt[now];
if(l == r) return ;
int mid = l + r >> 1;
ls[now] = ls[pre]; rs[now] = rs[pre];
if(key <= mid) insert(ls[pre],ls[now],l,mid,key);
else insert(rs[pre],rs[now],mid + 1,r,key);
}
int query(int t1,int t2,int t3,int t4,int l,int r,int k) {
if (l == r) return l;
int mid = l + r >> 1;
int dtl = cnt[ls[t1]] + cnt[ls[t2]] - cnt[ls[t3]] - cnt[ls[t4]];
if(k <= dtl) return query(ls[t1],ls[t2],ls[t3],ls[t4],l,mid,k);
else return query(rs[t1],rs[t2],rs[t3],rs[t4],mid + 1,r,k - dtl);
}
int root_num[maxn],siz[maxn], dep[maxn],dad[maxn][25];
void dfs(int x) {
siz[x] = 1;
for(int i = 1;i < 20;++i)
dad[x][i] = dad[dad[x][i - 1]][i - 1];
insert(root[dad[x][0]],root[x],1,MX,val[x]);
for(int i = head[x];i;i = edge[i].next) {
int v = edge[i].v;
if(v == dad[x][0]) continue;
root_num[v] = root_num[x];
dad[v][0] = x; dep[v] = dep[x] + 1;
dfs(v);
siz[x] += siz[v];
}
}
int LCA(int x,int y) {
if(dep[x] > dep[y]) std::swap(x,y);
for(int i = 19;i >= 0;i --) if(dep[dad[y][i]] >= dep[x]) y = dad[y][i];
if(x == y) return x;
for(int i = 19;i >= 0;i --) if(dad[y][i] != dad[x][i]) y = dad[y][i],x = dad[x][i];
return dad[x][0];
}
int work_q(int x,int y,int k) {
int lca = LCA(x,y);
int F_lca = dad[lca][0];
return query(root[x],root[y],root[lca],root[dad[lca][0]],1,MX,k);
}
void work_L(int x,int y) {
//printf("%d ***** %d\n",x,y);
if(siz[root_num[x]] < siz[root_num[y]]) std::swap(x,y);
dad[y][0] = x;
siz[root_num[x]] += siz[root_num[y]];
root_num[y] = root_num[x];
dep[y] = dep[x] + 1; add_edge(x,y);add_edge(y,x);
dfs(y);
}
int b[maxn];
void solve() {
tot = MX = 0;
n = read(),m = read(); t = read();
for(int i = 1;i <= n;++ i) b[i] = val[i] = read(),MX = std::max(val[i],MX);
std::sort(b + 1,b + n + 1);
int len = std::unique(b + 1,b + n + 1) - b - 1;
for(int i = 1;i <= n;++ i) val[i] = std::lower_bound(b + 1,b + len + 1,val[i]) - b,MX = std::max(val[i],MX);
for(int x,y,i = 1;i <= m;++ i) {
x = read(), y = read();
add_edge(x,y) , add_edge(y,x);
}
for(int i = 1;i <= n;++ i)
if(!root_num[i]) root_num[i] = i,dep[i] = 1,dfs(i);
char opt[20];
int lans = 0;
while(t --) {
scanf("%s",opt + 1) ;
if(opt[1] == 'Q') {
int x = read() ^ lans,y = read() ^ lans, k = read() ^ lans;
//printf("%d **** %d *** %d *** &&&\n",x,y,lans);
printf("%d\n",lans = b[work_q(x,y,k)]);
} else {
int x = read() ,y = read() ;
x ^= lans,y ^= lans;
//printf("%d **** %d *** %d *** ***\n",x,y,lans);
work_L(x,y);
}
}
}
int main() {
int testcase = read(); //whattttt the fucccck?
testcase = 1;
for(int i = 1;i <= testcase;++ i) solve();
}
luoguP3302 [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重建一下 注意 测试点 ...
- [BZOJ3123][Sdoi2013]森林 主席树+启发式合并
3123: [Sdoi2013]森林 Time Limit: 20 Sec Memory Limit: 512 MB Description Input 第一行包含一个正整数testcase,表示当 ...
- [SDOI2013]森林 主席树+启发式合并
这题的想法真的很妙啊. 看到题的第一眼,我先想到树链剖分,并把\(DFS\)序当成一段区间上主席树.但是会发现在询问的时候,可能会非常复杂,因为你需要把路径拆成很多条轻链和重链,它们还不一定连续,很难 ...
- 【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 ...
随机推荐
- 用C++的 new 代替 C 的 malloc 进行内存分配
例子: (int*)malloc(100*sizeof(int)) 是先取得int类型的字节宽度,然后乘100计算后得到400,然后调用malloc,并将400传递给函数,分配400字节的内存空间,但 ...
- 【PE结构】由浅入深PE基础学习-菜鸟手动查询导出表、相对虚拟地址(RVA)与文件偏移地址转换(FOA)
0 前言 此篇文章想写如何通过工具手查导出表.PE文件代码编程过程中的原理.文笔不是很好,内容也是查阅了很多的资料后整合出来的.希望借此加深对PE文件格式的理解,也希望可以对看雪论坛有所贡献.因为了解 ...
- canny 算子python实现
1. Canny介绍 Canny算子与Marr(LoG)边缘检测方法类似,也属于是先平滑后求导数的方法.John Canny研究了最优边缘检测方法所需的特性,给出了评价边缘检测性能优劣的三个指标: 1 ...
- vim常用命令总结 (转)【转】
转自:https://www.cnblogs.com/yangjig/p/6014198.html 在命令状态下对当前行用== (连按=两次), 或对多行用n==(n是自然数)表示自动缩进从当前行起的 ...
- dump_stack的简单使用 【转】
转自:http://blog.chinaunix.net/uid-26403844-id-3361770.html http://blog.csdn.net/ryfjx6/article/detail ...
- 命令行command line 使用 http proxy的设置方法 Setting Up HTTP Proxy in Terminal
Step 1: Install Shadowsocks Client Shadowsocks is an open-source proxy project to help people visit ...
- convertToNodeSpace和convertToWorldSpace ---实际应用
游戏中经常会用到一些变换: 游戏中武器和角色在一个layer上,为了效率,会考虑将bullet, effect和 PhysicsParticle分别放到不用的层上,对应的层上使用batchnode来提 ...
- GitHub上README.md的简单介绍
1.编辑README文件 大标题(一级标题):在文本下面加等于号,那么上方的文字就变成了大标题,等于号的个数无限制,但一定要大于0 大标题 ==== 中标题(二级标题):在文本下面加下划线,那么上方的 ...
- sqlserver服务启动后停止,传递给数据库 'master' 中的日志扫描操作的日志扫描号无效
电脑异常重启,导致SqlServer服务启动后,自动停止,在[计算机管理]-[事件查看器]-[windows日志]中进行查看系统错误日志,在[应用程序]下发现可能的错误信息有以下两条: 1.错误:传递 ...
- MyCP(课下作业,必做)
作业要求 编写MyCP.java 实现类似Linux下cp XXX1 XXX2的功能,要求MyCP支持两个参数: java MyCP -tx XXX1.txt XXX2.bin 用来把文本文件(内容为 ...