https://www.luogu.org/problemnew/show/P3377

左偏树+并查集

左偏树维护两个可合并的堆,并查集维护两个堆元素合并后可以找到正确的树根。

关键点在于删除一个堆的堆根的时候,需要把原来堆根的父指针指向新的堆根。这样并查集的性质就不会被破坏了。

#include<bits/stdc++.h>
using namespace std;
typedef long long ll; int solve(); int main() {
#ifdef Yinku
freopen("Yinku.in","r",stdin);
#endif // Yinku
solve();
} const int MAXN=100005;
int tot,v[MAXN],l[MAXN],r[MAXN],d[MAXN],f[MAXN];
//这其实是一片左偏树森林,用处不大
struct Leftist_Tree{
int Merge(int x,int y){
//将两棵左偏树,返回他们的新根
//当其中一棵是空树,直接返回另一棵
if(!x||!y)
return x+y;
//此处符号决定是最小堆还是最大堆
if(v[x]>v[y]||(v[x]==v[y]&&x>y))
swap(x,y);
//维持x不比y大,相等则把小的那个作为根
//把大数y插入小数x的右子树,所以是最小堆
r[x]=Merge(r[x],y);
if(r[x]==x)
r[x]=0;
//保证左子树比右子树高
//新的右子树的根的父亲是x,不需要可以删除
f[r[x]]=x;
if(d[l[x]]<d[r[x]])
swap(l[x],r[x]);
d[x]=d[r[x]]+1; return x;
}
int Build(int x){
//新建一棵只有x的左偏树,返回他的编号
tot++;
v[tot]=x;
//新树的节点的父亲为他本身,不需要可以删除
f[tot]=tot;
l[tot]=r[tot]=d[tot]=0;
return tot;
}
int Push(int x,int y){
//向编号为x的左偏树插入值为y的节点,其实就是新建只有y的左偏树然后合并
return Merge(x,Build(y));
}
int Top(int x){
//直接返回编号为x的左偏树的堆顶
return v[Get_root(x)];
}
int Get_root(int x){
//查找编号为x的节点所在的左偏树的根的序号,不需要可以删除
int r=x;
while(f[r]!=r)
r=f[r];
//路径压缩,直接指向他们的根
int k;
while(f[x]!=r){
k=f[x];
f[x]=r;
x=k;
}
return r;
}
int Pop(int x){
//将两个子树合并,相当于删除了堆顶
//return Merge(l[x],r[x]);
//当需要保留查询编号为x的节点在哪个子树时,不能够把堆顶丢掉
//把编号为x的元素删除
v[x]=-1;
//根x被删除,从x反过来指向他们的孩子就可以了
int rt=Merge(l[x],r[x]);
f[x]=f[l[x]]=f[r[x]]=rt;
return rt;
}
}; int solve() {
int n,m;
scanf("%d%d",&n,&m);
//printf("n,m=%d,%d\n",n,m);
Leftist_Tree lt;
f[0]=0;
v[0]=-1;
tot=0;
for(int i=1;i<=n;i++){
int x;
scanf("%d",&x);
//printf("x=%d\n",x);
lt.Push(i,x);
/*for(int j=1;j<=tot;j++){
printf("%d ",v[j]);
}
printf("\n");*/
}
for(int i=1;i<=m;i++){
int z,x,y;
scanf("%d",&z);
//printf("z=%d\n",z);
if(z==1){
scanf("%d%d",&x,&y);
//printf("x=%d y=%d\n",x,y);
if(v[x]==-1||v[y]==-1)
continue;
lt.Merge(lt.Get_root(x),lt.Get_root(y));
}
else{
scanf("%d",&x);
if(v[x]==-1){
printf("-1\n");
continue;
}
//printf("x=%d\n",x);
int fx=lt.Get_root(x);
printf("%d\n",lt.Top(fx));
lt.Pop(fx);
}
}
return 0;
}

洛谷 - P3377 - 【模板】左偏树(可并堆) - 左偏树 - 并查集的更多相关文章

  1. 洛谷 P3377 模板左偏树

    题目:https://www.luogu.org/problemnew/show/P3377 左偏树的模板题: 加深了我对空 merge 的理解: 结构体的编号就是原序列的位置. 代码如下: #inc ...

  2. 洛谷 P3377 【模板】左偏树(可并堆)

    洛谷 P3377 [模板]左偏树(可并堆) 题目描述 如题,一开始有N个小根堆,每个堆包含且仅包含一个数.接下来需要支持两种操作: 操作1: 1 x y 将第x个数和第y个数所在的小根堆合并(若第x或 ...

  3. 洛谷P3377 【模板】左偏树(可并堆) 题解

    作者:zifeiy 标签:左偏树 这篇随笔需要你在之前掌握 堆 和 二叉树 的相关知识点. 堆支持在 \(O(\log n)\) 的时间内进行插入元素.查询最值和删除最值的操作.在这里,如果最值是最小 ...

  4. 洛谷P3373 [模板]线段树 2(区间增减.乘 区间求和)

    To 洛谷.3373 [模板]线段树2 题目描述 如题,已知一个数列,你需要进行下面两种操作: 1.将某区间每一个数加上x 2.将某区间每一个数乘上x 3.求出某区间每一个数的和 输入输出格式 输入格 ...

  5. LCT总结——概念篇+洛谷P3690[模板]Link Cut Tree(动态树)(LCT,Splay)

    为了优化体验(其实是强迫症),蒟蒻把总结拆成了两篇,方便不同学习阶段的Dalao们切换. LCT总结--应用篇戳这里 概念.性质简述 首先介绍一下链剖分的概念(感谢laofu的讲课) 链剖分,是指一类 ...

  6. 洛谷P3834 [模板]可持久化线段树1(主席树) [主席树]

    题目传送门 可持久化线段树1(主席树) 题目背景 这是个非常经典的主席树入门题——静态区间第K小 数据已经过加强,请使用主席树.同时请注意常数优化 题目描述 如题,给定N个正整数构成的序列,将对于指定 ...

  7. 【AC自动机】洛谷三道模板题

    [题目链接] https://www.luogu.org/problem/P3808 [题意] 给定n个模式串和1个文本串,求有多少个模式串在文本串里出现过. [题解] 不再介绍基础知识了,就是裸的模 ...

  8. 「BZOJ2733」「洛谷3224」「HNOI2012」永无乡【线段树合并】

    题目链接 [洛谷] 题解 很明显是要用线段树合并的. 对于当前的每一个连通块都建立一个权值线段树. 权值线段树处理操作中的\(k\)大的问题. 如果需要合并,那么就线段树暴力合并,时间复杂度是\(nl ...

  9. BZOJ5291/洛谷P4458/LOJ#2512 [Bjoi2018]链上二次求和 线段树

    原文链接http://www.cnblogs.com/zhouzhendong/p/9031130.html 题目传送门 - LOJ#2512 题目传送门 - 洛谷P4458 题目传送门 - BZOJ ...

随机推荐

  1. c结构体里的数组与指针

    /* 訪问成员数组名事实上得到的是数组的相对地址.而訪问成员指针事实上是相对地址里的内容 */ struct buf_str { int length; char buf[0]; }; struct ...

  2. JavaScript中批量设置Css样式

    设置 input 元素的  属性: document.getElementsByTagName("INPUT")[0].setAttribute("属性",&q ...

  3. ETL Automation完整安装方法_(元数据存放在mysql数据库)

    安装前介质准备: DBI-1.636.tar.gz DBD-mysql-4.037.tar.gz ETL.tar mysql-5.6.12-linux-glibc2.5-x86_64.tar.gz P ...

  4. 多项式相乘快速算法原理及相应C代码实现---用到fft

    最近认真研究了一下算法导论里面的多项式乘法的快速计算问题,主要是用到了FFT,自己也实现了一下,总结如下. 1.多项式乘法 两个多项式相乘即为多项式乘法,例如:3*x^7+4*x^5+1*x^2+5与 ...

  5. java_类型强转

    class Father{ public void fromFather(){ System.out.println("fromFather"); } } interface in ...

  6. android检测当前网络是否可用

    在android程序中运行第一步就是检测当前有无可用网络  如果没有网络可用就退出程序  if (isConnect(this)==false)           {                ...

  7. PowerDesigner逆向工程,从SQL Server数据库生成Physical Model -----数据源方式

    1.File-Reverse Engineer-Database 2.DBMS选择SQL Server 3. 4. 5. 6. 7. 8. 9. 10. 11. 12. 13. 一路Next..... ...

  8. Open Source Streaming Server--EasyDarwin

    Welcome to EasyDarwin Streaming Server, which is an open source Streaming Server Based On Appple's D ...

  9. [自动化平台系列] - 初次使用 Macaca-前端自动化测试(1)

    1. 所先看一下官方地址,了解一下这个是不是你想要的测试工具 https://macacajs.github.io/macaca/environment-setup.html 2. 去掉sudo -- ...

  10. px dp 互转

    /** * * px dip 转换 */ public class DensityUtil { /** * 根据手机分辨率 dip 转 px */ static public int dip2px(C ...