http://www.lydsy.com/JudgeOnline/problem.php?id=3224

经典的平衡树模板题…各种平衡树好像都可以(黄学长之前好像还用vector卡过了这题)

所以这篇博客也就来存一下模板什么的…

如果发现有什么地方讲错的还请留言怼我


1.Treap

首先是经典的Treap:Treap=Tree+heap

这里每个结点有两个值$v$(结点的值)和$rnd$(一个随机值),叫Treap的原因也就是它遵循二叉查找树的性质($o$的左子树的$v$<$o$的$v$<$o$右子树的$v$)和堆的性质($o$左子树的$rnd$>$o$的$rnd$且$o$右子树的$rnd$>$o$的$rnd$,当然改成<也可以,不过这里我统一用>方便说)

二叉查找树的性质在插入的时候直接维护:如果要插入的比当前小就丢给左孩子,大的话就丢给右孩子(相等的话就直接存这个点上,这里用一个变量$w$来记录这个点实际上有几个点)

堆的性质通过旋转操作来维护:

比如如果你发现$o$的左孩子的$rnd$比$o$的$rnd$小了,它应该比较大才对,那这时候就可以把$o$右旋让它的左孩子变成它的父亲,可以发现这样子做二叉搜索树的性质并没有被破坏,如果是右孩子$rnd$比$o$大的话同理就用左旋啦,靠这两个操作就完成了堆性质的维护。因为用的是随机数所以期望情况下效率还是不错的。

(建议配合图片食用)

其他操作都比较简单了…具体还是见代码吧

#include<cstdio>
#include<cstdlib>
#define rep(i,n) for(register int i=1;i<=n;i++)
using namespace std;
inline int read()
{
int s=0,f=1;char c=getchar();
while(c<'0'||c>'9'){if(c=='-')f=-1;c=getchar();}
while(c>='0'&&c<='9'){s=s*10+c-'0';c=getchar();}
return s*f;
}
const int N=100005;
struct tree
{
int s,w,rnd,l,r,v;
}tr[N];
int n,rot,cnt,op;
inline void updata(int x)
{
tr[x].s=tr[tr[x].l].s+tr[tr[x].r].s+tr[x].w;
}
inline void rturn(int &x)
{
int t=tr[x].l;tr[x].l=tr[t].r;tr[t].r=x;
tr[t].s=tr[x].s;updata(x);x=t;
}
inline void lturn(int &x)
{
int t=tr[x].r;tr[x].r=tr[t].l;tr[t].l=x;
tr[t].s=tr[x].s;updata(x);x=t;
}
inline void insert(int &k,int val)
{
if(k==0)
{
k=++cnt;tr[k].v=val;tr[k].w=tr[k].s=1;
tr[k].rnd=rand();return;
}
tr[k].s++;
if(tr[k].v==val)
tr[k].w++;
else if(tr[k].v<val)
{
insert(tr[k].r,val);
if(tr[tr[k].r].rnd<tr[k].rnd)lturn(k);
}else
{
insert(tr[k].l,val);
if(tr[tr[k].l].rnd<tr[k].rnd)rturn(k);
}
}
inline void del(int &k,int val)
{
if(k==0)return;
if(tr[k].v==val)
{
if(tr[k].w>1)
tr[k].w--,tr[k].s--;
else
{
if(tr[k].l*tr[k].r==0)
k=tr[k].l+tr[k].r;
else if(tr[tr[k].l].rnd<tr[tr[k].r].rnd)
rturn(k),del(k,val);
else
lturn(k),del(k,val);
}
}else if(tr[k].v<val)
{
tr[k].s--;
del(tr[k].r,val);
}else
{
tr[k].s--;
del(tr[k].l,val);
}
}
inline int query_number_rank(int k,int val)
{
if(k==0)return 0;
if(tr[k].v==val)
return tr[tr[k].l].s+1;
else if(tr[k].v<val)
return query_number_rank(tr[k].r,val)+tr[tr[k].l].s+tr[k].w;
else
return query_number_rank(tr[k].l,val);
}
inline void query_rank(int k,int x,int &ans)
{
if(k==0)return;
if(x<=tr[tr[k].l].s)
query_rank(tr[k].l,x,ans);
else if(tr[tr[k].l].s+tr[k].w<x)
query_rank(tr[k].r,x-tr[tr[k].l].s-tr[k].w,ans);
else ans=tr[k].v;
}
inline void query_pre(int k,int x,int &ans)
{
if(k==0)return;
if(tr[k].v<x)
{
ans=tr[k].v;
query_pre(tr[k].r,x,ans);
}else
query_pre(tr[k].l,x,ans);
}
inline void query_sub(int k,int x,int &ans)
{
if(k==0)return;
if(tr[k].v>x)
{
ans=tr[k].v;
query_sub(tr[k].l,x,ans);
}else
query_sub(tr[k].r,x,ans);
}
int main()
{
srand(19260817);
n=read();
rep(i,n)
{
int x,ans=0;
op=read();x=read();
if(op==1)
insert(rot,x);
else if(op==2)
del(rot,x);
else if(op==3)
printf("%d\n",query_number_rank(rot,x));
else if(op==4)
{
query_rank(rot,x,ans);
printf("%d\n",ans);
}else if(op==5)
{
query_pre(rot,x,ans);
printf("%d\n",ans);
}else
{
query_sub(rot,x,ans);
printf("%d\n",ans);
}
}
return 0;
}

2.01Trie

别人博客的:http://www.cnblogs.com/KingSann/articles/7339563.html

有个神奇的东西叫01Trie…其实就是一颗字符集为$\{0,1 \}$的Trie。

然后我们把要存的变量搞成二进制丢到这颗Trie里面就行了!非常好写。

缺点就是就是空间比较大,以及如果要存浮点数就有点无能为力了…(能不能写成$a*10^b$的形式丢树上呢…脑补ing…)

#include<cstdio>
const int N=100005*33;
const int T=(int)1e7;//偏移的大小
const int B=31;
inline int read()
{
int s=0,f=1;char c=getchar();
while(c<'0'||c>'9'){if(c=='-')f=-1;c=getchar();}
while(c>='0'&&c<='9'){s=s*10+c-'0';c=getchar();}
return s*f;
}
int n,cnt=1,x,op;
int tr[N][2],f[N];
inline void insert(int val,int c)
{
int k=1;val+=T;
for(register int i=B;~i;i--)
{
int t=(val>>i)&1;
if(!tr[k][t])tr[k][t]=++cnt;
k=tr[k][t];f[k]+=c;
}
}
inline int rank(int val)
{
int k=1,res=0;val+=T;
for(register int i=B;~i;i--)
{
int t=(val>>i)&1;
if(t)res+=f[tr[k][0]];
k=tr[k][t];
}
return res;
}
inline int kth(int val)
{
int k=1,res=0;
for(register int i=B;~i;i--)
{
if(val>f[tr[k][0]])res|=(1<<i),val-=f[tr[k][0]],k=tr[k][1];
else k=tr[k][0];
}
res-=T;return res;
}
int main()
{
n=read();
while(n--)
{
op=read();x=read();
if(op==1)insert(x,1);
else if(op==2)insert(x,-1);
else if(op==3)printf("%d\n",rank(x)+1);
else if(op==4)printf("%d\n",kth(x));
else if(op==5)printf("%d\n",kth(rank(x)));
else if(op==6)printf("%d\n",kth(rank(x+1)+1));
}
return 0;
}

先写到这了其他的有空再更…

[日常摸鱼]bzoj3224普通平衡树-Treap、Splay、01Trie、替罪羊树…的更多相关文章

  1. 简析平衡树(一)——替罪羊树 Scapegoat Tree

    前言 平衡树在我的心目中,一直都是一个很高深莫测的数据结构.不过,由于最近做的题目的题解中经常出现"平衡树"这三个字,我决定从最简单的替罪羊树开始,好好学习平衡树. 简介 替罪羊树 ...

  2. BZOJ 3224 - 普通平衡树 - [Treap][Splay]

    题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id=3224 Description 您需要写一种数据结构(可参考题目标题),来维护一些数,其中 ...

  3. [洛谷P3369] 普通平衡树 Treap & Splay

    这个就是存一下板子...... 题目传送门 Treap的实现应该是比较正经的. 插入删除前驱后继排名什么的都是平衡树的基本操作. #include<cstdio> #include< ...

  4. BZOJ3224普通平衡树【Splay】

    3224: Tyvj 1728 普通平衡树 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 11751  Solved: 5013 Descriptio ...

  5. [日常摸鱼]Luogu1801 黑匣子(NOI导刊)

    题意:写一个数据结构,要求滋兹两种操作,ADD:插入一个数,GET:令$i++$然后输出第$i$小的数 这个数据结构当然是平衡树啦!(雾) 写个Treap直接过掉啦- #include<cstd ...

  6. [bzoj3224]普通平衡树[Treap]

    Treap 的各种操作,模板题,要再写几遍 #include <iostream> #include <algorithm> #include <cstdio> # ...

  7. Hash 日常摸鱼笔记

    本篇文章是Hash在信息学竞赛中的应用的学习笔记,分多次更新(已经有很多坑了) 一维递推 首先是Rabin-Karp,对于一个长度为\(m\)的串\(S\) \(f(S)=\sum_{i=1}^{m} ...

  8. [日常摸鱼]HDU1724 Ellipse-自适应Simpson法

    模板题~ QAQ话说Simpson法的原理我还是不太懂-如果有懂的dalao麻烦告诉我~ 题意:每次给一个椭圆的标准方程,求夹在直线$x=l$和$x=r$之间的面积 Simpson法 (好像有时候也被 ...

  9. [日常摸鱼]bzoj1257余数之和

    题意:输入$k,n$,求$\sum_{i=1}^n k \mod i$ $k \mod i=k-i*\lfloor \frac{k}{i} \rfloor $,$n$个$k$直接求和,后面那个东西像比 ...

随机推荐

  1. 用过MindManager后才知道思维导图原来这么简单

    哈喽大家好!时间过得真是太快了,一眨眼这一年就接近尾声了,相信我们都度过了不平凡但十分充足的一年,不知道大家在2020年中有没有令自己满意的收获呢? 相信大家各自都有精彩的收获,我们不妨把它们总结一下 ...

  2. Boom 3D带你聆听《我们的乐队》音乐盛宴

    说到前段时间大热的音乐类综艺节目,<我们的乐队>肯定值得一提.<我们的乐队>是由谢霆锋.萧敬腾.王俊凯担任导师,以团队的形式组成各种风格的乐队,并通过同场比拼,最终选出获胜的乐 ...

  3. python 几个循环的效率测试

    前言:对于我这种追求极致的人来说,效率很重要. 前面看到网上关于python循环的测评,到自己在项目中的应用,发现,并不是这么回事.所以,写下次博文,一次性了解这个问题. 语言版本:python3.6 ...

  4. JAVA 中的Optional (臭名昭著的空指针异常(NullPointerException))

    从 Java 8 引入的一个很有趣的特性是 Optional  类.Optional 类主要解决的问题是臭名昭著的空指针异常(NullPointerException) -- 每个 Java 程序员都 ...

  5. Snap Build Your Own Block修炼之道-添加自定义类别

    Snap Build Your Own Block自我修炼方法:1.所有的面向对象,其实是对面向过程的抽象过程而已: 2.面对别人的开源项目时,需要找准源头(即项目运行的起点,当然有的是没有的哈,没有 ...

  6. C语言项目——工程化编程的案例分析

    一.VSCode安装及环境配置 初始在Win下安装Mingw-w64/GCC 和 GDB,在VSCode下打开项目案例,发现在linktable中需要包含pthread头文件.此文件是基于Linux系 ...

  7. 安装seafile记录文档

    安装yum-cron.iptables .关闭selinux yum -y install cronie yum -y install yum-cron systemctl start yu,-cro ...

  8. 『CDN』让你的网站访问起来更加柔顺丝滑

    我是风筝,公众号「古时的风筝」,一个兼具深度与广度的程序员鼓励师,一个本打算写诗却写起了代码的田园码农! 文章会收录在 JavaNewBee 中,更有 Java 后端知识图谱,从小白到大牛要走的路都在 ...

  9. 通过Dbeaver创建表格的时候,设置主键

    通过Dbeaver创建表格的时候,设置主键 Dbeaver介绍: ​ 这是一个开源的数据库连接工具,你需要安装jre才可以使用这个软件 在使用Dbeaver创建表的时候,会发现,不能直观地设置主键 这 ...

  10. PyQt(Python+Qt)学习随笔:树型部件QTreeWidget中的项编辑方法editTriggers、editItem和openPersistentEditor作用及对比分析

    老猿Python博文目录 专栏:使用PyQt开发图形界面Python应用 老猿Python博客地址 在树型部件QTreeWidget中,有三种方法触发进行项数据的编辑:editTriggers触发编辑 ...