洛谷-P3369-普通平衡树(Treap)
题目传送门
标题说平衡树,那么应该AVL,红黑树都能过,但是这次做这题主要是学习Treap,所以花了几天搞出了这题。其他方法以后再说吧
- Treap(带旋转)
#include <bits/stdc++.h>
using namespace std;
const int MAXN = ;
const int INF = 0x3f3f3f3f;
struct Treap {
int son[];
int val, rand;
int cnt, size;
} node[MAXN];
int root, tot;
void update(int i) {
int ls = node[i].son[];
int rs = node[i].son[];
node[i].size = node[ls].size + node[rs].size + node[i].cnt;
}
void rotate(int& r, int index) {
int k = node[r].son[index ^ ];
node[r].son[index ^ ] = node[k].son[index];
node[k].son[index] = r;
update(r); update(k);
r = k;
}
void insert(int& r, int v) {
if (r == ) {
r = ++ tot;
node[r].val = v;
node[r].rand = rand();
node[r].size = node[r].cnt = ;
return;
}
if (node[r].val == v) {
node[r].size ++;
node[r].cnt ++;
return;
}
int index = v > node[r].val;
insert(node[r].son[index], v);
if (node[node[r].son[index]].rand < node[r].rand) rotate(r, index ^ );
update(r);
}
void erase(int& r, int v) {
if (r == ) return;
if (v == node[r].val) {
if (node[r].cnt > ) {
node[r].cnt --;
node[r].size --;
return;
}
if (node[r].son[] == || node[r].son[] == ) {
r = node[r].son[] + node[r].son[];
} else {
int ls = node[r].son[];
int rs = node[r].son[];
int index = node[ls].rand < node[rs].rand;
rotate(r, index);
erase(node[r].son[index], v);
}
} else {
int index = v > node[r].val;
erase(node[r].son[index], v);
}
update(r);
}
int get_rank(int r, int v) {
int size = node[node[r].son[]].size;
int cnt = node[r].cnt;
if (node[r].val == v) return size + ;
if (node[r].val > v) return get_rank(node[r].son[], v);
if (node[r].val < v) return size + cnt + get_rank(node[r].son[], v);
}
int get_val(int r, int rk) {
int size = node[node[r].son[]].size;
int cnt = node[r].cnt;
if (rk < size + ) return get_val(node[r].son[], rk);
if (rk >= size + && rk <= size + cnt) return node[r].val;
if (rk > size + cnt) return get_val(node[r].son[], rk - size - cnt);
}
int get_pre(int r, int v) {
if (r == ) return -INF;
if (node[r].val >= v) return get_pre(node[r].son[], v);
else return max(node[r].val, get_pre(node[r].son[], v));
}
int get_suc(int r, int v) {
if (r == ) return INF;
if (node[r].val <= v) return get_suc(node[r].son[], v);
else return min(node[r].val, get_suc(node[r].son[], v));
}
int main() {
srand(time(NULL));
int n; scanf("%d", &n);
for (int i = ; i <= n; i++) {
int opt, x;
scanf("%d%d", &opt, &x);
if (opt == ) insert(root, x);
if (opt == ) erase(root, x);
if (opt == ) printf("%d\n", get_rank(root, x));
if (opt == ) printf("%d\n", get_val(root, x));
if (opt == ) printf("%d\n", get_pre(root, x));
if (opt == ) printf("%d\n", get_suc(root, x));
}
return ;
}学到的第二个用随机数的算法,所以这个Treap的高度也是有的随机的。但是起码不太可能退化成链状。
- Treap(无旋转)
#include <bits/stdc++.h>
using namespace std;
const int MAXN = ;
const int INF = 0x3f3f3f3f;
struct Treap {
int ls, rs;
int val, rand;
int size;
} node[MAXN];
int tot, root;
int add_node(int v) {
int i = ++tot;
node[i].val = v;
node[i].rand = rand();
node[i].ls = node[i].rs = ;
node[i].size = ;
return i;
}
void split(int rt, int& a, int& b, int v) {
if (rt == ) {
a = b = ;
return;
}
if (node[rt].val <= v) {
a = rt;
split(node[rt].rs, node[a].rs, b, v);
} else {
b = rt;
split(node[rt].ls, a, node[b].ls, v);
}
node[rt].size = node[node[rt].ls].size + node[node[rt].rs].size + ;
}
void merge(int& rt, int a, int b) {
if (a == || b == ) {
rt = a + b;
return;
}
if (node[a].rand < node[b].rand) {
rt = a;
merge(node[rt].rs, node[a].rs, b);
} else {
rt = b;
merge(node[rt].ls, a, node[b].ls);
}
node[rt].size = node[node[rt].ls].size + node[node[rt].rs].size + ;
}
void insert(int& rt, int v) {
int x = , y = ;
split(rt, x, y, v);
merge(x, x, add_node(v));
merge(rt, x, y);
}
void erase(int&rt, int v) {
int x = , y = , z = ;
split(rt, x, z, v);
split(x, x, y, v - );
merge(y, node[y].ls, node[y].rs);
merge(x, x, y);
merge(rt, x, z);
}
int get_rank(int rt, int v) {
int x = , y = ;
split(rt, x, y, v - );
int rk = node[x].size + ;
merge(rt, x, y);
return rk;
}
int get_val(int rt, int rk) {
int size = node[node[rt].ls].size;
if (rk < size + ) return get_val(node[rt].ls, rk);
if (rk == size + ) return node[rt].val;
if (rk > size + ) return get_val(node[rt].rs, rk - size - );
}
int get_pre(int rt, int v) {
int x = , y = ;
split(rt, x, y, v - );
int pre = get_val(x, node[x].size);
merge(rt, x, y);
return pre;
}
int get_suc(int rt, int v) {
int x = , y = ;
split(rt, x, y, v);
int suc = get_val(y, );
merge(rt, x, y);
return suc;
}
int main() {
srand(time(NULL));
int n; scanf("%d", &n);
for (int i = ; i <= n; i++) {
int opt, x;
scanf("%d%d", &opt, &x);
if (opt == ) insert(root, x);
if (opt == ) erase(root, x);
if (opt == ) printf("%d\n", get_rank(root, x));
if (opt == ) printf("%d\n", get_val(root, x));
if (opt == ) printf("%d\n", get_pre(root, x));
if (opt == ) printf("%d\n", get_suc(root, x));
}
return ;
}昨天看懂了带旋转的Treap,今天来写无旋转的Treap,感觉确实是比带旋转的好敲而且好懂,关键就是理解split()和merge()两个核心函数。
洛谷-P3369-普通平衡树(Treap)的更多相关文章
- [洛谷P3369] 普通平衡树 Treap & Splay
这个就是存一下板子...... 题目传送门 Treap的实现应该是比较正经的. 插入删除前驱后继排名什么的都是平衡树的基本操作. #include<cstdio> #include< ...
- 洛谷P3369普通平衡树(Treap)
题目传送门 转载自https://www.cnblogs.com/fengzhiyuan/articles/7994428.html,转载请注明出处 Treap 简介 Treap 是一种二叉查找树.它 ...
- 洛谷P3369 普通平衡树
刚学平衡树,分别用了Splay和fhq-treap交了一遍. 这是Splay的板子,貌似比较短? Splay #include <iostream> #include <cstdio ...
- 洛谷P3369 【模板】普通平衡树(Treap/SBT)
洛谷P3369 [模板]普通平衡树(Treap/SBT) 平衡树,一种其妙的数据结构 题目传送门 题目描述 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作: 插入x数 删除 ...
- 【洛谷P3369】【模板】普通平衡树题解
[洛谷P3369][模板]普通平衡树题解 题目链接 题意: 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作:1. 插入x数2. 删除x数(若有多个相同的数,因只删除一个)3 ...
- [洛谷P3391] 文艺平衡树 (Splay模板)
初识splay 学splay有一段时间了,一直没写...... 本题是splay模板题,维护一个1~n的序列,支持区间翻转(比如1 2 3 4 5 6变成1 2 3 6 5 4),最后输出结果序列. ...
- 绝对是全网最好的Splay 入门详解——洛谷P3369&BZOJ3224: Tyvj 1728 普通平衡树 包教包会
平衡树是什么东西想必我就不用说太多了吧. 百度百科: 一个月之前的某天晚上,yuli巨佬为我们初步讲解了Splay,当时接触到了平衡树里的旋转等各种骚操作,感觉非常厉害.而第二天我调Splay的模板竟 ...
- BZOJ3223/洛谷P3391 - 文艺平衡树
BZOJ链接 洛谷链接 题意 模板题啦~2 代码 //文艺平衡树 #include <cstdio> #include <algorithm> using namespace ...
- BZOJ3224/洛谷P3391 - 普通平衡树(Splay)
BZOJ链接 洛谷链接 题意简述 模板题啦~ 代码 //普通平衡树(Splay) #include <cstdio> int const N=1e5+10; int rt,ndCnt; i ...
- 洛谷 P3391 文艺平衡树
题目描述 您需要写一种数据结构(可参考题目标题),来维护一个有序数列,其中需要提供以下操作:翻转一个区间,例如原有序序列是5 4 3 2 1,翻转区间是[2,4]的话,结果是5 2 3 4 1 --b ...
随机推荐
- Delphi流的操作_文件合并
unit Unit1; interface uses Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms ...
- 201803-1 跳一跳 Java
思路: 一个变量plus记录叠加的数,遇到2就+2 import java.util.Scanner; public class Main { public static void main(Stri ...
- bootstrap 基础表单
表单中常见的元素主要包括:文本输入框.下拉选择框.单选按钮.复选按钮.文本域和按钮等.其中每个控件所起的作用都各不相同,而且不同的浏览器对表单控件渲染的风格都各有不同. ☑ LESS版本:对应源文 ...
- c语言:自增自减运算符的操作详解
博主在回忆c语言的基本知识时,突然发现自增自减运算符(--.++)这个知识点有些模糊不清,故博主为了给同为小白的同学们提供一些经验,特写下这篇文章. 首先,自增自减运算符共有两种操作方式. 比如,我先 ...
- Python说文解字_杂谈07
1. 深入dict from collections.abc import Mapping,MutableMapping # dict 属于mapping类型 a = {} print(isinsta ...
- Python说文解字_杂谈04
1. 鸭子类型: 当你看到一只鸟走来像鸭子,游泳起来像鸭子,叫起来也像鸭子,他么他就可以叫做鸭子.任何可迭代的对象.一样的方法,可以用可迭代的话,就可以迭代的组合打印.__getitem__可以塞到任 ...
- Python笔记_第一篇_面向过程_第一部分_1.Python环境的设置(含虚拟机)
*Python环境的设置 Python的环境设置也就是所需工作平台的软件搭建.常用的搭建平台IOS系统+Linux系统和Windows+Linux系统的搭建.这里主要说明的是Windows+Linux ...
- subprocess.Popen stdout重定向内容实时获取
python 打开一个新进程执行系统命令, test 执行完才能获取返回, test1 实时获取返回结果 import subprocess def test(cmd): p = subprocess ...
- 吴裕雄--天生自然Linux操作系统:Linux 安装
Linux 的安装,安装步骤比较繁琐,现在其实云服务器挺普遍的,价格也便宜,如果直接不想搭建,也可以直接买一台学习用用,参考各大云服务器比较:https://www.runoob.com/linux/ ...
- win32框架
win32的框架 1.入口函数 2.窗口注册类信息 3.窗口创建 4.显示窗口 5.更新窗口 6.消息循环 7.入口函数结束 WNDCLASSEX wcex;窗口类结构 wcex.cbSize = s ...