bzoj 2733 平衡树启发式合并
首先对于一个连通块中,询问我们可以直接用平衡树来求出排名,那么我们可以用并查集来维护各个块中的连通情况,对于合并两个平衡树,我们可以暴力的将size小的平衡树中的所有节点删掉,然后加入大的平衡树中,因为每个点只可能被删除插入logn次,所以时间复杂度为nlog^2n。
/**************************************************************
Problem: 2733
User: BLADEVIL
Language: C++
Result: Accepted
Time:2112 ms
Memory:64868 kb
****************************************************************/
//By BLADEVIL
#include <cstdio>
#define maxn 100010
#define maxt 4000010
using namespace std;
int n,m;
int father[maxn],a[maxn],root[maxn];
int left[maxt],right[maxt],size[maxt],key[maxt];
int tot,save;
int adr[maxn];
int getfather(int x)
{
if (father[x]==x) return x;
return father[x]=getfather(father[x]);
}
void swap(int &x,int &y)
{
int z=x;
x=y; y=z;
}
void left_rotate(int &t)
{
int k=right[t];
right[t]=left[k];
left[k]=t;
size[k]=size[t];
size[t]=size[left[t]]+size[right[t]]+;
t=k;
}
void right_rotate(int &t)
{
int k=left[t];
left[t]=right[k];
right[k]=t;
size[k]=size[t];
size[t]=size[left[t]]+size[right[t]]+;
t=k;
}
void maintain(int &t,bool flag)
{
if (!flag)
{
if (size[left[left[t]]]>size[right[t]])
right_rotate(t); else
if (size[right[left[t]]]>size[right[t]])
left_rotate(left[t]),right_rotate(t); else return;
} else
{
if (size[right[right[t]]]>size[left[t]])
left_rotate(t); else
if (size[left[right[t]]]>size[left[t]])
right_rotate(right[t]),left_rotate(t); else return;
}
maintain(left[t],); maintain(right[t],);
maintain(t,); maintain(t,);
}
void t_insert(int &t,int v)
{
if (!t)
{
t=++tot;
left[t]=right[t]=;
key[t]=v;
size[t]=;
} else
{
size[t]++;
if (v>key[t]) t_insert(right[t],v); else t_insert(left[t],v);
maintain(t,v>key[t]);
}
}
int t_delete(int &t,int v)
{
size[t]--;
if ((key[t]==v)||((v>key[t])&&(!right[t]))||((v<key[t])&&(!left[t])))
{
save=key[t];
if ((!left[t])||(!right[t]))
t=left[t]+right[t]; else key[t]=t_delete(left[t],v+);
} else
return (v>key[t])?t_delete(right[t],v):t_delete(left[t],v);
}
int t_rank(int &t,int k)
{
if (size[left[t]]+==k) return key[t];
return (k<=size[left[t]])?t_rank(left[t],k):t_rank(right[t],k-size[left[t]]-);
}
void combine(int x,int y)
{
int fx,fy;
fx=getfather(x); fy=getfather(y);
if (size[root[fx]]<size[root[fy]]) swap(fx,fy);
father[fy]=fx;
if (fx!=fy)
{
fy=root[fy];
while (root[fy])
{
t_insert(root[fx],a[fy]);
t_delete(root[fy],a[fy]);
}
}
}
void ask(int x,int k)
{
x=root[getfather(x)];
if (size[x]>=k) printf("%d\n",adr[t_rank(x,k)]); else printf("-1\n");
}
void init()
{
int x,y;
scanf("%d%d",&n,&m);
for (int i=;i<=n;i++) scanf("%d",&a[i]);
for (int i=;i<=n;i++) adr[a[i]]=i;
for (int i=;i<=n;i++) father[i]=i,t_insert(root[i],a[i]);
while (m--)
{
scanf("%d%d",&x,&y);
combine(x,y);
}
}
void solve()
{
int test,x,y;
char s[];
scanf("%d",&test);
while (test--)
{
scanf("%s%d%d",&s,&x,&y);
if (s[]=='B')
combine(x,y); else ask(x,y);
}
}
int main()
{
init();
solve();
return ;
}
bzoj 2733 平衡树启发式合并的更多相关文章
- bzoj 2733 Splay 启发式合并,名次树
题意:给定一个带点权的无向图,有两种操作: 1.将两个连通分量合并. 2.查询某个连通分量里的第K大点. 题解: 用并查集维护连通关系,一开始建立n棵splay树,然后不断合并,查询. 处理技巧: 1 ...
- BZOJ 2809: [Apio2012]dispatching( 平衡树 + 启发式合并 )
枚举树上的每个结点做管理者, 贪心地取其子树中薪水较低的, 算出这个结点为管理者的满意度, 更新答案. 用平衡树+启发式合并, 时间复杂度为O(N log²N) ------------------- ...
- ☆ [HNOI2012] 永无乡 「平衡树启发式合并」
题目类型:平衡树启发式合并 传送门:>Here< 题意:节点可以连边(不能断边),询问任意两个节点的连通性与一个连通块中排名第\(k\)的节点 解题思路 如果不需要询问排名,那么并查集即可 ...
- bzoj 2809 左偏树\平衡树启发式合并
首先我们对于一颗树,要选取最多的节点使得代价和不超过m,那么我们可以对于每一个节点维护一个平衡树,平衡树维护代价以及代价的和,那么我们可以在logn的时间内求出这个子树最多选取的节点数,然后对于一个节 ...
- BZOJ 2733 & splay的合并
题意: 带权联通块,添边与查询联通块中第k大. SOL: splay合并+并查集. 我以为splay可以用奇技淫巧来简单合并...调了一下午终于幡然醒悟...于是就只好一个一个慢慢插...什么启发式合 ...
- 【pb_ds】【平衡树启发式合并】【并查集】bzoj2733 [HNOI2012]永无乡
用并查集维护联通性.对每个联通块维护一个平衡树.合并时启发式合并.比较懒,用了pb_ds. #include<cstdio> #include<ext/pb_ds/assoc_con ...
- 【BZOJ1483】[HNOI2009]梦幻布丁(平衡树启发式合并+并查集)
题目: BZOJ1483 分析: (这题码了一下午,码了近250行,但是意外跑的比本校各位神仙稍快,特写博客纪念) 首先能看出一个显然的结论:颜色段数只会变少不会变多. 我们考虑用并查集维护区间,对于 ...
- bzoj 1483 链表 + 启发式合并
思路:将颜色相同的建成一个链表, 变颜色的时候进行链表的启发式合并.. 因为需要将小的接到大的上边,所以要用个f数组. #include<bits/stdc++.h> #define LL ...
- bzoj 1483 链表启发式合并
首先我们可以比较容易的在n的时间内算出来开始的答案,我们维护一些链表,分别表示不同的颜色,那么我们在计算答案的时候,只需要扫一遍所有的链表,判断链表相邻两项是否在序列中相邻,不相邻的话肯定在这其中的一 ...
随机推荐
- TCP系列05—连接管理—4、TCP连接的ISN、连接建立超时及TCP的长短连接
一.TCP连接的ISN 之前我们说过初始建立TCP连接的时候的系列号(ISN)是随机选择的,那么这个系列号为什么不采用一个固定的值呢?主要有两方面的原因 防止同一个连接的不同实例(di ...
- Spring Boot(四)@EnableXXX注解分析
在学习使用springboot过程中,我们经常碰到以@Enable开头的注解,其实早在Spring3中就已经出现了类似注解,比如@EnableTransactionManagement.@ Enabl ...
- mysql中删除重复记录,并保留重复数据中的一条数据的SQL语句
正好想写一条删除重复语句并保留一条数据的SQL,网上查了一部分资料写的很详细,但还是在这里写下自己的理解,以遍后续学习 .如下: 表字段和数据: SQL语句: [sql] view plain cop ...
- CCS3 动画-鼠标放上去放大背景图片
---〉 效果如上,一个简单的过渡放大效果, <!DOCTYPE HTML> <html> <body> <style> #test{ width:30 ...
- JavaScript Array 类型
除 Object类型外,Array 类型算是ECMAScript中最常用的类型了,而且,ECMAScript中的数组和其他多数语言的数组有着非常大的差别,虽然这些数组都是数据的有序列表,但与其他语言不 ...
- openstack之Glance介绍
什么是Glance glance即image service(镜像服务),是为虚拟机的创建提供镜像服务 为什么要有Glance 我们基于openstack是构建基本的Iaas平台对外提供虚机,而虚机在 ...
- BZOJ4770 图样(概率期望+动态规划)
考虑求出所有MST的权值和再除以方案数,方案数显然是2mn. 按位考虑,显然应该让MST里的边高位尽量为0.那么根据最高位是0还是1将点集划分成两部分,整张图的MST就是由两部分各自的MST之间连一条 ...
- vue-cli项目打包出现空白页和路径错误问题
vue-cli项目打包: 1. 命令行输入:npm run build 打包出来后项目中就会多了一个文件夹dist,这就是我们打包过后的项目. 第一个问题,文件引用路径.我们直接运行打包后的文件夹 ...
- JS判断当前DOM树是否加载完毕
/** * @function Monitor whether the document tree is loaded. * @param fn */function domReady(fn) { i ...
- 中国MOOC_面向对象程序设计——Java语言_第2周 对象交互_秒计时的数字时钟
第2周编程题 查看帮助 返回 第2周编程题,在课程所给的时钟程序的基础上修改 依照学术诚信条款,我保证此作业是本人独立完成的. 温馨提示: 1.本次作业属于Online Judge题目,提交后由系 ...