左偏树

左偏树

一种可以合并的堆

前置知识

dist

对于一棵二叉树,我们定义 外节点 为左儿子或右儿子为空的节点,定义一个外节点的 为 ,一个不是外节点的节点 为其到子树中最近的外节点的距离加一。空节点的dist为0。

那么左偏树就是一颗满足堆的性质的二叉树,它的左儿子的dist大于等于右儿子的

核心

核心操作是合并,合并到时候,要满足堆得性质,对于小根堆,就把较小的根作为新的根节点,然后递归合并右儿子和另外一个堆。如果不满足左偏性质,还要交换一下儿子。

int merge(int x,int y){
    if(!x||!y) return x+y;
    if(el[y]<el[x]) swap(x,y);
    el[x].r=merge(el[x].r,y);
    if(dis[el[x].l]<dis[el[x].r]) swap(el[x].l,el[x].r);
    dis[x]=dis[el[x].r]+1;
    return x;
}

插入

把一个单独的节点看成一个堆

删除堆顶

把堆顶的左右儿子合并,并且处理一下信息.

为了快速知道堆顶是谁,我们需要搞一个并查集,并且维护他,删除堆顶的时候要

rt[el[x].l]=rt[el[x].r]=rt[x]=merge(el[x].l,el[x].r);

因为我们要路径压缩,不然太慢了

删除任意元素

删除的时候首先合并左右儿子,把左右儿子形成的新堆合并后接到爹上。

注意要维护dist,可以选择(上面可以)把dist大的当成左儿子,然后合并,这样还不用交换了。

维护整体操作

和线段树差不多,打标记,合并就可以

模板题

#include<cstdio>
#include<iostream>
#include<cstring>
#include<iomanip>
#include<cmath>
#include<stack>
#include<algorithm>
using namespace std;
template<class T>inline void read(T &x)
{
    x=0;register char c=getchar();register bool f=0;
    while(!isdigit(c))f^=c=='-',c=getchar();
    while(isdigit(c))x=(x<<3)+(x<<1)+(c^48),c=getchar();
    if(f)x=-x;
}
template<class T>inline void print(T x)
{
    if(x<0)putchar('-'),x=-x;
    if(x>9)print(x/10);
    putchar('0'+x%10);
}
int n;
int m;
struct e{
    int v;
    int id;
    int l;
    int r;
    friend bool operator <(e x,e y){
        return x.v==y.v?x.id<y.id:x.v<y.v;
    }
}el[100005];
int dis[100005];
int rt[100005];
int f;
int x,y;
int del[100005];

int find(int x){
    return x==rt[x]?x:rt[x]=find(rt[x]);
}
int merge(int x,int y){
    if(!x||!y) return x+y;
    if(el[y]<el[x]) swap(x,y);
    el[x].r=merge(el[x].r,y);
    if(dis[el[x].l]<dis[el[x].r]) swap(el[x].l,el[x].r);
    dis[x]=dis[el[x].r]+1;
    return x;
}
int main(){
    read(n);read(m);
    for(int i=1;i<=n;++i){
        read(el[i].v);
        el[i].id=i;
        rt[i]=i;
    }
    for(int i=1;i<=m;++i){
        read(f);
        if(f==1){
            read(x);read(y);
            int xx=find(x);
            int yy=find(y);
            if(del[x]||del[y]||xx==yy){
                continue;
            }
            rt[xx]=rt[yy]=merge(xx,yy);
        }else{
            read(x);
            if(del[x]){
                cout<<"-1n";continue;
            }

            x=find(x);
            del[x]=1;
            cout<<el[x].v<<endl;
            rt[el[x].l]=rt[el[x].r]=rt[x]=merge(el[x].l,el[x].r);
            el[x].l=el[x].r=dis[x]=0;
        }
    }
    return 0;
}
暂无评论

发送评论 编辑评论


				
|´・ω・)ノ
ヾ(≧∇≦*)ゝ
(☆ω☆)
(╯‵□′)╯︵┴─┴
 ̄﹃ ̄
(/ω\)
∠( ᐛ 」∠)_
(๑•̀ㅁ•́ฅ)
→_→
୧(๑•̀⌄•́๑)૭
٩(ˊᗜˋ*)و
(ノ°ο°)ノ
(´இ皿இ`)
⌇●﹏●⌇
(ฅ´ω`ฅ)
(╯°A°)╯︵○○○
φ( ̄∇ ̄o)
ヾ(´・ ・`。)ノ"
( ง ᵒ̌皿ᵒ̌)ง⁼³₌₃
(ó﹏ò。)
Σ(っ °Д °;)っ
( ,,´・ω・)ノ"(´っω・`。)
╮(╯▽╰)╭
o(*////▽////*)q
>﹏<
( ๑´•ω•) "(ㆆᴗㆆ)
😂
😀
😅
😊
🙂
🙃
😌
😍
😘
😜
😝
😏
😒
🙄
😳
😡
😔
😫
😱
😭
💩
👻
🙌
🖕
👍
👫
👬
👭
🌚
🌝
🙈
💊
😶
🙏
🍦
🍉
😣
Source: github.com/k4yt3x/flowerhd
颜文字
Emoji
小恐龙
花!
上一篇
下一篇