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

因为博主精力和实力有限,学不懂 fhq treap 了,因此只介绍 leafy tree 解法

leafy tree 的本质是一颗平衡线段树,它的根节点保存整颗树的信息,是不会变的,因此可以高效的实现可持久化

#include <bits/stdc++.h>
#define update(u) if(u -> left -> size) u -> size = u -> left -> size + u -> right -> size, u -> value = u -> right -> value
#define new_Node(a, b, c, d) (&(t[cnt++] = Node(a, b, c, d)))
#define merge(a, b) new_Node(a -> size + b -> size, b -> value, a, b)
#define ratio 4
using namespace std; const int N = 500000 + 10;
const int logN = 20; struct Node {
int size, value;
Node *left, *right;
Node () {}
Node (int a, int b, Node *c, Node *d) : size(a), value(b), left(c), right(d) {}
}*root[N], *null, t[N * logN * 11 / 10]; int n, cnt = 0; Node *maintain(Node *u) {
Node *cur = new_Node(u -> size, u -> value, u -> left, u -> right);
if(cur -> left -> size > cur -> right -> size * ratio) cur -> left = maintain(cur -> left), cur -> right = maintain(cur -> right), cur -> right = merge(cur -> left -> right, cur -> right), cur -> left = cur -> left -> left;
if(cur -> right -> size > cur -> left -> size * ratio) cur -> left = maintain(cur -> left), cur -> right = maintain(cur -> right), cur -> left = merge(cur -> left, cur -> right -> left), cur -> right = cur -> right -> right;
return cur;
} Node *ins(Node *u, int x) {
Node *cur = new_Node(u -> size, u -> value, u -> left, u -> right);
if(cur -> size == 1) cur -> left = new_Node(1, min(cur -> value, x), null, null), cur -> right = new_Node(1, max(cur -> value, x), null, null);
else if(x > cur -> left -> value) cur -> right = ins(cur -> right, x); else cur -> left = ins(cur -> left, x);
update(cur); return cur;
} Node *earse(Node *u, int x) {
Node *cur = new_Node(u -> size, u -> value, u -> left, u -> right);
if(u -> size == 1 && u -> value != x) return cur;
if(cur -> left -> size == 1 && x == cur -> left -> value) *cur = *cur -> right;
else if(cur -> right -> size == 1 && x == cur -> right -> value) *cur = *cur -> left;
else if(x > cur -> left -> value) cur -> right = earse(cur -> right, x); else cur -> left = earse(cur -> left, x);
update(cur); return cur;
} int find(Node *u, int x) {
if(u -> size == 1) return u -> value;
return x > u -> left -> size ? find(u -> right, x - u -> left -> size) : find(u -> left, x);
} int Rank(Node *u, int x) {
// printf("u -> value = %d, x = %d\n", u -> value, x);
if(u -> size == 1) return 1;
return x > u -> left -> value ? Rank(u -> right, x) + u -> left -> size : Rank(u -> left, x);
} int main() {
scanf("%d", &n);
null = new Node(0, 0, 0, 0);
root[0] = new Node(1, INT_MAX, null, null);
for(int i = 1; i <= n; i++) {
int a, t, pre;
scanf("%d %d %d", &pre, &t, &a);
if(t == 1) root[i] = maintain(ins(root[pre], a));
if(t == 2) root[i] = maintain(earse(root[pre], a));
if(t == 3) printf("%d\n", Rank(root[pre], a)), root[i] = root[pre];
if(t == 4) printf("%d\n", find(root[pre], a)), root[i] = root[pre];
if(t == 5) {
int k = Rank(root[pre], a) - 1;
if(k == 0) puts("-2147483647");
else printf("%d\n", find(root[pre], k));
root[i] = root[pre];
}
if(t == 6) {
int k = Rank(root[pre], a + 1);
if(k == root[pre] -> size) puts("2147483647");
else printf("%d\n", find(root[pre], k));
root[i] = root[pre];
}
}
return 0;
}

关于新建节点时写

#define new_Node(a, b, c, d) (&(*st[cnt++] = Node(a, b, c, d)))

#define new_Node(a, b, c, d) (&(t[cnt++] = Node(a, b, c, d)))

的区别

leafy tree 实现可持久化平衡树的时候不能高效的垃圾回收,第一种就变成废物了,第二种在可持久化时更加高效

关于旋转的时候写

Node *maintain(Node *u) {
Node *cur = new_Node(u -> size, u -> value, u -> left, u -> right);
if(cur -> left -> size > cur -> right -> size * ratio) cur -> left = maintain(cur -> left), cur -> right = maintain(cur -> right), cur -> right = merge(cur -> left -> right, cur -> right), st[--cnt] = cur -> left, cur -> left = cur -> left -> left;
else if(cur -> right -> size > cur -> left -> size * ratio) cur -> left = maintain(cur -> left), cur -> right = maintain(cur -> right), cur -> left = merge(cur -> left, cur -> right -> left), st[--cnt] = cur -> right, cur -> right = cur -> right -> right;
return cur;
}

Node *maintain(Node *u) {
Node *cur = new_Node(u -> size, u -> value, u -> left, u -> right);
if(cur -> left -> size > cur -> right -> size * ratio) cur -> right = merge(cur -> left -> right, cur -> right), cur -> left = cur -> left -> left;
if(cur -> right -> size > cur -> left -> size * ratio) cur -> left = merge(cur -> left, cur -> right -> left), cur -> right = cur -> right -> right;
return cur;
}

的区别

第一种情况需要在插入和删除的时候调用 maintain,是整棵树平衡,比较正常

第二种情况在每次 update 之后 maintain,在可持久化时不能保证全局平衡,可能不太优秀?(这玩意是个玄学

关于 merge 的高效实现(因为博主太菜了就咕咕咕了

luoguP3835 [模板]可持久化平衡树的更多相关文章

  1. 洛谷.3835.[模板]可持久化平衡树(fhq treap)

    题目链接 对每次Merge(),Split()时产生的节点都复制一份(其实和主席树一样).时间空间复杂度都为O(qlogq).(应该更大些 因为rand()?内存真的爆炸..) 对于无修改的操作实际上 ...

  2. luogu P3919 [模板]可持久化数组(可持久化线段树/平衡树)(主席树)

    luogu P3919 [模板]可持久化数组(可持久化线段树/平衡树) 题目 #include<iostream> #include<cstdlib> #include< ...

  3. Luogu P3835 【模板】可持久化平衡树(fhq Treap)

    P3835 [模板]可持久化平衡树 题意 题目背景 本题为题目普通平衡树的可持久化加强版. 题目描述 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作(对于各个以往的历史版本 ...

  4. [Luogu 3835]【模板】可持久化平衡树

    Description 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作(对于各个以往的历史版本): 插入x数 删除x数(若有多个相同的数,因只删除一个,如果没有请忽略该操作 ...

  5. 洛谷P3835 【模板】可持久化平衡树

    题目背景 本题为题目 普通平衡树 的可持久化加强版. 数据已经经过强化 感谢@Kelin 提供的一组hack数据 题目描述 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作( ...

  6. P3835 【模板】可持久化平衡树

    题目描述 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作(对于各个以往的历史版本): 插入x数 删除x数(若有多个相同的数,因只删除一个,如果没有请忽略该操作) 查询x数的 ...

  7. LG3835 【模板】可持久化平衡树

    题意 您需要写一种数据结构(可参考题目标题),来维护一些数,其中需要提供以下操作(对于各个以往的历史版本): 插入x数 删除x数(若有多个相同的数,因只删除一个,如果没有请忽略该操作) 查询x数的排名 ...

  8. 2021.07.02 P1383 高级打字机题解(可持久化平衡树)

    2021.07.02 P1383 高级打字机题解(可持久化平衡树) 分析: 从可以不断撤销并且查询不算撤销这一骚操作可以肯定这是要咱建一棵可持久化的树(我也只会建可持久化的树,当然,还有可持久化并查集 ...

  9. 可持久化Trie & 可持久化平衡树 专题练习

    [xsy1629]可持久化序列 - 可持久化平衡树 http://www.cnblogs.com/Sdchr/p/6258827.html [bzoj4260]REBXOR - Trie 事实上只是一 ...

随机推荐

  1. python's twenty day for me 继承 和 super()方法

    super(): 在单继承中就是单纯的寻找父类. 在多继承中就是根据子节点所在图 的mro顺序,找寻下一个类. 遇到多继承和super(): 对象.方法 1,找到这个对象对应的类. 2,将这个类的所有 ...

  2. ulimit open files linux打开文件数设置验证

    #include <stdio.h> #include <sys/types.h> #include <fcntl.h> #include <stdlib.h ...

  3. Java如何解决form表单上传文件,以及页面返回处理结果通知!

    前端JSP代码 <form id='formSumbit' class='form-horizontal' action='/ncpay/route/chlsubmcht/batchImpor' ...

  4. MySQL备份还原之一mydumper

    1)源码编译安装 1.下载 mydumper源码 2.解压 [mysql@localhost ~]$ tar -xvf mydumper-0.9.1.tar mydumper-0.9.1/CMakeL ...

  5. Java中包、类、方法、属性、常量的命名规则

    1:包(package):用于将完成不同功能的类分门别类,放在不同的目录(包)下,包的命名规则:将公司域名反转作为包名.比如www.baidu.com 对于包名:每个字母都需要小写.比如:com.ba ...

  6. windows下利用VMware安装mac:构建…

    安装条件: 硬件:一台拥有支持虚拟技术的64位双核处理器和2GB以上内存的PC. 注意:运行MAC OS,需要电脑支持虚拟技术(VT),安装时,需要将VT启动,在BIOS中开启. 关于如何检测你的电脑 ...

  7. 7. Reverse Integer 反转整数

    [抄题]: 将一个整数中的数字进行颠倒,当颠倒后的整数溢出时,返回 0 (标记为 32 位整数).   样例 给定 x = 123,返回 321 给定 x = -123,返回 -321 [暴力解法]: ...

  8. SQL 数据排重,去掉重复数据 有用

    .最大的错误:    在对数据排重的时候,首先想到的就是Distinct,虽然这很管用,但多数场合下不适用,因为通常排重后还要做进一步处理,比如对编号排重后要按日期统计等. 无法排重的Group by ...

  9. Oracle——分组函数

    AVG(平均值)和 SUM (合计)函数 可以对数值型数据使用AVG 和 SUM 函数. AVG组函数忽略空值 --在组函数中使用NVL函数 --求平均值 )) MIN(最小值)和 MAX(最大值)函 ...

  10. Codeforces 429B B. Working out

    题目意思: 给n*m的矩阵,每个格子有个数,A从(1,1)出发只能向下或右走,终点为(n,m),B从(n,1)出发只能向上或右走,终点为(1,m).两个人的速度不一样,走到的格子可以获的该格子的数,两 ...