首页 > 其他分享 >平衡树专栏

平衡树专栏

时间:2023-07-21 18:45:18浏览次数:36  
标签:pre cnt return int updata 专栏 平衡 size

普通平衡树【Treap】

平衡树可以实现很多操作,而且时间都是在 \(O(log_n)\) 级别的

  • 芝士复杂度

为什么可以稳定在 \(O(log_n)\) 呢?是因为它不仅是二叉查找树,而且还是一个堆(堆的值是用随机函数实现的),所以它的深度就不会被卡,稳定在 \(O(log_n)\) 的深度,所以时间也就是 \(O(log_n)\) 了。

  • 芝士操作

那么 \(treap\) 都包括哪些操作呢? 插入,删除,根据编号找值,根据值找编号,找前驱和后继,由此可见其也是非常的强悍

  • 芝士实现

1.如果它插入或者删除之后不符合二叉查找树的性质了怎么办,那就
旋分为左旋和右旋,具体来说左旋就是把当前节点和它的两个儿子逆时针旋转一下,再计算值,右旋相反。这是一个非常重要的操作。
2.如何插入?如果插入的值在原树上已经有了,那我们就可以直接加进去,如果没有就新开一个节点给他弄进树,如果当前节点不是要找的插入位置那就根据情况判断是往左查找还是往右。
3.如何删除?删除的操作其实和插入大差不差,但是多了的操作,在删除节点之前一下,将要删除的点到叶子节点,这样就会让删除更简单!这里可以画一张图结合下面代码仔细想想。
4.如何根据编号找值?我们只需要在树中再开一个数组记录以它为根的这棵树的 \(size\),然后再通过编号和树的 \(size\) 作比较,然后向下递归直到找到要找的节点就好。
5.如何根据值找编号?因为我们的这棵树是二叉查找树,所以查找就像普通二叉查找树的查找一样(其实和删除操作也差不多),向下递归就好!
6.如何找前驱和后继?前驱就是比当前节点的值小的值里的最大的值,后继就是比当前节点大的值里最小的值,我们既然是二叉查找树,那这还不简单,前驱只要在它的左子树一直向右找到叶子节点就好,后继相反。

到现在为止,您就已经学会 \(treap\) 了!

  • 芝士代码
#include <iostream>
#include <cstdlib>

using namespace std;
const int Max = 1e5+10;
const int inf = 0x7fffffff;

int n , root , tot;

struct zx{
	int l , r , size , cnt , dat , pre;
}a[Max];

int add(int pre) {
	tot ++;
	a[tot].pre = pre;
	a[tot].dat = rand();
	a[tot].cnt = a[tot].size = 1;
	return tot;
}

void updata(int x) {
	a[x].size = a[a[x].l].size + a[a[x].r].size + a[x].cnt;
}

void build() {
	root = add(-inf);
	a[root].r = add(inf);
	updata(root);
}

void zig(int &p) {
	int q = a[p].l;
	a[p].l = a[q].r;
	a[q].r = p;
	p = q;
	updata(a[p].r) , updata(p);
}

void zag(int &p) {
	int q = a[p].r;
	a[p].r = a[q].l;
	a[q].l = p;
	p = q;
	updata(a[p].l) , updata(p);
}

void insert(int &x , int pre) {
	if(x == 0) {
		x = add(pre);
		return ;
	}
	if(a[x].pre == pre) {
		a[x].cnt ++;
		updata(x);
		return;
	}
	if(pre < a[x].pre) {
		insert(a[x].l , pre);
		if(a[a[x].l].dat > a[x].dat) zig(x);
	}
	else {
		insert(a[x].r , pre);
		if(a[a[x].r].dat > a[x].dat) zag(x);
	}
	updata(x);
}

void delet(int &x , int pre) {
	if(x == 0) return;
	if(a[x].pre == pre) {
		if(a[x].cnt > 1) {
			a[x].cnt --;
			updata(x);
			return;
		}
		if(a[x].l || a[x].r) {
			if(!a[x].r || a[a[x].l].dat > a[a[x].r].dat) zig(x) , delet(a[x].r , pre);
			else zag(x) , delet(a[x].l , pre);
			updata(x);
		}
		else x = 0;
	}
	if(pre > a[x].pre) delet(a[x].r , pre);
	else delet(a[x].l , pre);
	updata(x);
}

int get_pre(int p, int rank) {
	if (p == 0) return 0;
	if (a[a[p].l].size >= rank) return get_pre(a[p].l, rank);
	if (a[a[p].l].size + a[p].cnt >= rank) return a[p].pre;
	return get_pre(a[p].r, rank - a[p].cnt - a[a[p].l].size); 
}

int get_rank(int x, int pre) {
	if(x == 0) return 0;
	if(a[x].pre == pre) return a[a[x].l].size + 1;
	if(a[x].pre > pre) return get_rank(a[x].l , pre);
	return get_rank(a[x].r , pre) + a[a[x].l].size + a[x].cnt;
}

int get_front(int pre) {
	int ans = 0 , x = root;
	while(x) {
		if(a[x].pre >= pre) x = a[x].l;
		else ans = a[x].pre , x = a[x].r;
	}
	return ans;
}

int get_next(int pre) {
	int ans = 0 , x = root;
	while(x) {
		if(a[x].pre <= pre) x = a[x].r;
		else ans = a[x].pre , x = a[x].l;
	}
	return ans;
}

signed main() {
	build();
	cin >> n;
	for(long long i = 1; i <= n; i++) {
		long long opt , x;
		cin >> opt >> x;
		if(opt == 1) insert(root , x);
		else if(opt == 2) delet(root , x);
		else if(opt == 3) cout << get_rank(root , x) - 1 << endl;
		else if(opt == 4) cout << get_pre(root, x + 1) << endl;
		else if(opt == 5) cout << get_front(x) << endl;
		else cout << get_next(x) << endl;
	}
	return 0;
}

标签:pre,cnt,return,int,updata,专栏,平衡,size
From: https://www.cnblogs.com/roselu/p/17572204.html

相关文章

  • 【项目实战】Kafka 重平衡 Consumer Group Rebalance 机制
    ......
  • uniapp专栏 —— vscode报错 'uni' is not defined.
    写在前面这些内容基于通过cli搭建的uniapp项目,使用了vite4,ts4.9,vue3(组合式API,setup语法糖)。如果有版本不一致,请谨慎参考。正文uni是一个全局变量,但是eslint没有识别到。避免这个错误报错在.eslintrc.js文件中加上配置globals:{uni:true},......
  • 数据结构--二叉平衡树
    二叉平衡树回顾:二叉排序树的查找二叉排序树的不平衡会影响查找效率,所有我们要尽量让二叉树的形态均衡.AVL树(平衡二叉树)必须是二叉排序树左子树和右子树的高度之差的绝对值小于等于1左子树和右子树也是平衡二叉排序树平衡因子该结点左子树与右子树的高度差.平......
  • BST(二叉搜索树)、AVL(平衡二叉树)、RBT(红黑树)的区别
    一、二叉搜索树(BST:BinarySortTree)二叉查找树就是左结点小于根节点,右结点大于根节点的一种排序树,也叫二叉搜索树。二叉查找树比普通树查找更快,查找、插入、删除的时间复杂度为O(logN)。但是二叉查找树有一种极端的情况,就是会变成一种线性链表似的结构。此时时间复......
  • JVM专栏-内存分配与回收策略
    对象的内存分配,就是在堆上分配(也可能经过JIT编译后被拆散为标量类型并间接在栈上分配),对象主要分配在新生代的Eden区上,少数情况下可能直接分配在老年代,分配规则不固定,取决于当前使用的垃圾收集器组合以及相关的参数配置。以下列举几条最普遍的内存分配规则,供大家学习。对象优......
  • JVM专栏-类文件结构
    JVM的“无关性”谈论JVM的无关性,主要有以下两个:平台无关性:任何操作系统都能运行Java代码语言无关性:JVM能运行除Java以外的其他代码Java源代码首先需要使用Javac编译器编译成.class文件,然后由JVM执行.class文件,从而程序开始运行。JVM只认识.class文件,......
  • JVM专栏-类加载的时机
    类加载的时机类的生命周期类从被加载到虚拟机内存开始,到卸载出内存为止,它的整个生命周期包括以下7个阶段:加载验证准备解析初始化使用卸载验证、准备、解析3个阶段统称为连接。加载、验证、准备、初始化和卸载这5个阶段的顺序是确定的,类的加载过程必须按照这种......
  • JVM专栏-垃圾回收器
    本文以HotSpot虚拟机为例,讲述一下几种常见的垃圾回收器.新生代垃圾收集器Serial垃圾收集器(单线程)只开启一条GC线程进行垃圾回收,并且在垃圾收集过程中停止一切用户线程,即StopTheWorld。一般客户端应用所需内存较小,不会创建太多对象,而且堆内存不大,因此垃圾收集器回收......
  • JVM专栏-垃圾回收策略与算法
    程序计数器、虚拟机栈、本地方法栈随线程而生,也随线程而灭;栈帧随着方法的开始而入栈,随着方法的结束而出栈。这几个区域的内存分配和回收都具有确定性,在这几个区域内不需要过多考虑回收的问题,因为方法结束或者线程结束时,内存自然就跟随着回收了。而对于Java堆和方法区,我们只有在......
  • 平衡树 - 知识点梳理
    _本文中一行代码都没有,因为平衡树的代码变化很大,关键是理解思想,代码根本不重要,需要代码可以看我的提交记录_平衡树是一种优化过的BST。BST由于没有保证深度因此每次查询复杂度最坏为\(O(n)\),而平衡树通过在中序遍历不改变的情况下对树的结构做一些适当的调整,使每次查询时间复杂......