很多人觉得可持久化treap很慢,但是事实上只是他们可持久化treap的写法不对。他们一般是用split和merge实现所有功能,但是这样会有许多不必要的分裂。其实我们可以用一种特殊的方式来实现插入和删除。

插入:我们先随机出新建节点的Rank值,随二叉查找树的顺序找到第一个Rank比新建节点Rank小的节点,将以这个节点为根的子树按Key值分裂成两颗树并作为新建节点的左子树和右子树。

删除:我们用二叉查找树的方式找到删除节点,释放节点空间并将节点左子树和右子树合并代替原树。

由于随机构建二叉查找树从每个节点到叶节点期望距离是O(1)的,所以在插入删除中期望合并的树的深度是O(1)的。这样一来插入删除的log常数就只受查找速度影响(貌似比普通treap还快)

 #include<cstdio>
#include<cstdlib>
#include<algorithm>
using namespace std ; struct Node {
int Key ;
int Rank ;
int Size ;
Node * L ;
Node * R ;
Node ( int , int , int ) ;
void maintain () ;
} ; Node * Nil = new Node ( , INT_MIN , ) ; Node :: Node ( const int Key = , const int Rank = ( rand () ) , const int Size = ) :
Key ( Key ) , Rank ( Rank ) ,
Size ( Size ) , L ( Nil ) , R ( Nil ) {} void Node :: maintain () {
Size = L -> Size + + R -> Size ;
} Node * Merge ( Node * const L , Node * const R ) {
if ( L == Nil ) return R ;
else if ( R == Nil ) return L ;
else if ( L -> Rank > R -> Rank ) {
L -> R = Merge ( L -> R , R ) ;
L -> maintain () ;
return L ;
} else {
R -> L = Merge ( L , R -> L ) ;
R -> maintain () ;
return R ;
}
} typedef pair < Node * , Node * > Npair ; void Split1 ( Node * const O , const int K ,
Node * & L , Node * & R ) {
if ( O == Nil ) L = R = Nil ;
else if ( O -> L -> Size <= K ) {
Split1 ( O -> L , K , L , R ) ;
O -> L = R ;
R = O ;
O -> maintain () ;
} else {
Split1 ( O -> R , K - ( O -> L -> Size + ) , L , R ) ;
O -> R = L ;
L = O ;
O -> maintain () ;
}
} void Split2 ( Node * const O , const int Key ,
Node * & L , Node * & R ) {
if ( O == Nil ) L = R = Nil ;
else if ( Key <= O -> Key ) {
Split2 ( O -> L , Key , L , R ) ;
O -> L = R ;
R = O ;
O -> maintain () ;
} else {
Split2 ( O -> R , Key , L , R ) ;
O -> R = L ;
L = O ;
O -> maintain () ;
}
} int GetRank ( const Node * O , const int Key ) {
int ans = ;
while ( O != Nil ) {
if ( O -> Key <= Key ) {
O = O -> L ;
} else {
ans += O -> L -> Size + ;
O = O -> R ;
}
}
return ans ;
} void Erase ( Node * & O , const int Key ) {
if ( O == Nil ) return ;
else if ( O -> Key == Key ) O = Merge ( O -> L , O -> R ) ;
else if ( Key < O -> Key ) {
Erase ( O -> L , Key ) ;
O -> maintain () ;
} else {
Erase ( O -> R , Key ) ;
O -> maintain () ;
}
} void Insert ( Node * & O , const int Key , const int Rank = rand () ) {
if ( O -> Rank < Rank ) {
Node * const np = new Node ( Key ) ;
Split2 ( O , Key , np -> L , np -> R ) ;
( O = np ) -> maintain () ;
} else if ( Key < O -> Key ) {
Insert ( O -> L , Key , Rank ) ;
O -> maintain () ;
} else {
Insert ( O -> R , Key , Rank ) ;
O -> maintain () ;
}
}

高rong效chang的可持久化treap的更多相关文章

  1. UVALive 6145 Version Controlled IDE(可持久化treap、rope)

    题目链接:https://icpcarchive.ecs.baylor.edu/index.php?option=com_onlinejudge&Itemid=8&page=show_ ...

  2. BZOJ 3595: [Scoi2014]方伯伯的Oj SBT+可持久化Treap

    3595: [Scoi2014]方伯伯的Oj Time Limit: 6 Sec  Memory Limit: 256 MBSubmit: 102  Solved: 54[Submit][Status ...

  3. 【模板】可持久化文艺平衡树-可持久化treap

    题目链接 题意 对于各个以往的历史版本实现以下操作: 在第 p 个数后插入数 x . 删除第 p 个数. 翻转区间 [l,r],例如原序列是 \(\{5,4,3,2,1\}\),翻转区间 [2,4] ...

  4. 可持久化Treap

    终于写了一次可持久化Treap,做的是可持久化序列的模板题. Treap Treap=Tree+Heap,是一个随机化的数据结构.它的每个节点至少有两个关键字,一个是我们要存储的\(val\),一个是 ...

  5. Codeforces - 38G 可持久化Treap 区间操作

    题意:\(n\)个人排队,每个人有重要度\(p\)和不要脸度\(c\),如果第\(i\)个人的重要度大于第\(i-1\)个人的重要度,那么他们之间可以交换,不要脸度-1,交换后先前的第\(i\)个人也 ...

  6. Codeforces - 675D 可持久化Treap 树形操作

    题意:模拟二叉树的构造过程,给出\(n\)个节点,每次从根插入,小于当前节点转到左儿子,否则右儿子,输出第\([2,n]\)个节点的父亲的权值 直接手动模拟会被链式结构T掉 网上找了下发现二叉树的性质 ...

  7. 平衡树与可持久化treap

    平衡树(二叉树) 线段树不支持插入or删除一个数于是平衡树产生了 常见平衡树:treap(比sbt慢,好写吧),SBT(快,比较好写,有些功能不支持),splay(特别慢,复杂度当做根号n来用,功能强 ...

  8. 脑洞大开加偏执人格——可持久化treap版的Link Cut Tree2

    试了一下先上再下的Treap方式,很高兴,代码变短了,但是,跑的变慢了!!!其实慢得不多,5%左右.而且这个版本的写法不容易写错..只要会一般可持久化Treap的人写着都不难...就是相对于(压行的) ...

  9. 脑洞大开加偏执人格——可持久化treap版的Link Cut Tree

    一直没有点动态树这个科技树,因为听说只能用Splay,用Treap的话多一个log.有一天脑洞大开,想到也许Treap也能从底向上Split.仔细思考了一下,发现翻转标记不好写,再仔细思考了一下,发现 ...

随机推荐

  1. Nginx入门篇(六)之反向代理和负载均衡

    一.Nginx负载均衡集群 介绍 负载均衡(Load Balance)集群提供了一种行之有效的办法,来扩展网络设备和服务器负载.带宽和吞吐量,同时加强了网络数据处理能力,提供了网络的灵活性和可用性. ...

  2. Elasticsearch6.2集群搭建

    Elasticsearch6.2集群搭建 2018年04月02日 11:07:45 这个名字想了很久 阅读数:14154    版权声明:本博客为学习.笔记之用,以笔记形式记录学习的知识与感悟.学习过 ...

  3. HTML基本代码教学,第三天

    HTML 今天由于个人情况,身体不适,但是为了大家的学习进度,咱们以纯文字得形式来简单了解下今天的学习内容 今儿咱们来了解下表单 <form id=" "  name=&qu ...

  4. 动态权限<一>基本介绍

    android 6.0以上为了保护用户的隐私,和以往被人诟病的权限机制,确立了新的权限机制.从 Android 6.0(API 级别 23)开始,用户开始在应用运行时向其授予权限,而不是在应用安装时授 ...

  5. 「日常训练」Jongmah(Codeforces-1110D)

    题意 你有n个数字,范围[1, m],你可以选择其中的三个数字构成一个三元组,但是这三个数字必须是连续的或者相同的,每个数字只能用一次,问这n个数字最多构成多少个三元组? 分析 根据官方Editori ...

  6. 记录阿里云ECS(Centos7.4)安装mysql 8.0.X服务

    #*.rpm介绍 大多数二进制rpm包都包含在名称中倒数第二个字段中编译rpm的体系结构..rpm软件包有那么几种 *.src.rpm 源程序包,要先通过编译才能安装 *.noarch.rpm 该包适 ...

  7. Qt 利用XML文档,写一个程序集合 一

    接到领导需求安排,说公司未来的硬件设备会越来越多,与每个设备对应的设备检测和设置程序也会增多.导致软甲太多,不好掌控.所以希望做一个完整的软件,但是呢,每个子程序还得独立,应为每个用户购买的设备不是一 ...

  8. Qt-QML-Canvas写个小小的闹钟

    先看下演示效果 大致过程 先绘制仪表盘,圆圈和刻度 剩下再绘制三个指针 最后在绘制上面的电子时钟 下面写源代码 import QtQuick 2.0 Rectangle { id:root ancho ...

  9. 廖雪峰git笔记

    查看本地机子的在Git上的名字和邮箱:git config user.namegit config user.email 对所有仓库指定相同的用户名和Email地址:git config --glob ...

  10. CentOS7使用阿里源安装最新版Docker

    卸载已经安装的Docker sudo yum remove docker \ docker-client \ docker-client-latest \ docker-common \ docker ...