AVL树 - 学习笔记
2017-08-29 14:35:55
writer:pprp
AVL树就是带有平衡条件的二叉查找树。每个节点的左子树和右子树高度相差最多为1的二叉查找树
空树的高度定为-1
对树的修正称为旋转
对内部的来说是双旋,对外部的调整修正是单旋
----------------------------------------------------------------------------------------------------------------
由于一次旋转总能解决问题,因此编写非递归程序要比编写递归程序快很多,但是非递归方式编写比较难
还是很多人都选择递归的方式,这里也选择递归的方式,比较容易理解;
代码如下:(还没有完全写好,删除部分不太理解)
/*
@theme:AVL tree
@writer:pprp
@begin:14:32
@end:16:26
@declare:带有平衡条件的二叉查找树,这里不需要一个创建节点的函数,
因为相关操作已经在Insert函数中完成了
@date:2017/8/29
*/ #include <bits/stdc++.h> using namespace std;
struct AvlNode;
typedef struct AvlNode* Position;
typedef struct AvlNode* AvlTree; AvlTree MakeEmpty(AvlTree T);
Position Find(int X, AvlTree T);
Position FindMin(AvlTree T);
Position FindMax(AvlTree T);
AvlTree Insert(int X, AvlTree T);
AvlTree Delete(int X, AvlTree T); static Position SingleRotateWithLeft(Position);
static Position DoubleRotateWithLeft(Position);
static Position SingleRotateWithRight(Position);
static Position DoubleRotateWithRight(Position); //AVL tree节点声明
struct AvlNode
{
int element;
AvlTree left;
AvlTree right;
int Height;
}; //一个快速的函数来返回节点高度
static int Height(Position P)
{
if(P == NULL)
return -;
else
return P->Height;
} //向AVL树中插入节点的函数
AvlTree Insert(int X, AvlTree T)
{
//如果节点为空,建立然后返回一个单节点的树
if(T == NULL)
{
T = new AvlNode();
//如果失败
if(T == NULL)
{
cout << "Out of place!" << endl;
}
else
{
T->element = X;
T->Height = ;
T->left = T->right = NULL;
}
}
else if(X < T->element) //如果头结点不为空
{
T->left = Insert(X,T->left);
//question
//如果不满足AVL tree的要求了,且左侧高于右侧,对左侧进行处理
if(Height(T->left) - Height(T->right) == )
{
if(X < T->left->element)//如果小于左边
T = SingleRotateWithLeft(T);//进行左侧的单旋
else
T = DoubleRotateWithLeft(T);//进行左侧的双旋
}
}
else if(X > T->element)
{
T->right = Insert(X, T->right);
//如果右侧高于左侧进行旋转
if(Height(T->right) - Height(T->left) == )
{
if(X > T->right->element)
T = SingleRotateWithRight(T);
else
T = DoubleRotateWithRight(T);
}
}
//else X is in the tree already , we'll do nothing
T->Height = max(Height(T->left),Height(T->right)) + ;
return T;
} //执行单旋转的左边 LL
static Position SingleRotateWithLeft(Position K2)
{
Position K1;
K1 = K2->left;
K2->left = K1->right;
K1->right = K2; K2->Height = max(Height(K2->left),Height(K2->right))+;
K1->Height = max(Height(K1->left),Height(K1->right))+; return K1;
} //执行双旋转的左边 LR
static Position DoubleRotateWithLeft(Position K3)
{
//rotate between K1 and K2
K3->left = SingleRotateWithRight(K3->left);
//rotate between K3 and K2
return SingleRotateWithLeft(K3);
} //执行单旋转的右边 RR
static Position SingleRotateWithRight(Position K1)
{
Position K2; K2 = K1->left;
K1->left = K2->right;
K2->right = K1; K1->Height = max(Height(K1->right),Height(K2->left))+;
K2->Height = max(Height(K2->right),Height(K2->left))+; return K2;
}
//执行双旋转的右边 RL
static Position DoubleRotateWithRight(Position K3)
{
K3->right = SingleRotateWithLeft(K3->right);
return SingleRotateWithRight(K3);
} //进行中序遍历
void MidPrint(AvlTree T)
{
if(T != NULL)
{
MidPrint(T->left);
cout << T->element << " ";
MidPrint(T->right);
}
} //查找函数,返回一个指针
Position Find(int X, AvlTree T)
{
if(T == NULL)
return NULL;
if(X < T->element)
return Find(X, T->right);
else if(X > T->element)
return Find(X, T->left);
else
return T;
} //找到最大值
Position FindMax(AvlTree T)
{
if(T == NULL)
return NULL;
else if(T->right == NULL)
return T;
else
return FindMax(T->right);
} //找到最小值
Position FindMin(AvlTree T)
{
if(T == NULL)
return NULL;
else if(T->left == NULL)
return T;
else
return FindMin(T->left);
} //删除节点,返回根节点
AvlTree Delete(int X, AvlTree T)
{
AvlTree tmp = Find(X,T);
if(T == NULL || tmp == NULL)
return NULL;
if(X < T->element)//如果在左子树中
{
T->left = Delete(X,T->left);
//开始调整由于删除带来的影响
if(Height(T->right) - Height(T->left) == )
{
AvlTree K = T->right;//因为右边高度比较高
if(Height(K->left) > Height(K->right))
DoubleRotateWithRight(T);
else
SingleRotateWithRight(T);
}
}
else if(X > T->element) //如果在右子树中
{
T->right = Delete(X,T->right);
if(Height(T->left) - Height(T->right) == )
{
AvlTree K = T->left;
if(Height(K->right) > Height(K->left))
DoubleRotateWithLeft(T);
else
SingleRotateWithLeft(T);
}
}
else if(X == T->element)
{
// 如果两个孩子非空
if(T->left && T->right)
{
//维护AVL树特性:本来可以选择两种方式进行删除
//现在要优先删除高度比较高的子树
if(Height(T->left) > Height(T->right))
{
AvlTree Max = FindMax(T->left); //找到最大值
T->element = Max->element; //将内容进行替换
delete(Max); //删除该节点
Max = NULL;
}
else
{
AvlTree Min = FindMin(T->right);
T->element = Min->element;
delete(Min);//删除该节点
Min = NULL;
}
}
else //如果有一个子节点或者没有子节点
{
AvlTree tmp = T;
T = T->left == NULL ? T->right:T->left;
delete(tmp);
}
}
return T;
} //清空整个树
AvlTree MakeEmpty(AvlTree T)
{
if(T!= NULL)
{
MakeEmpty(T->right);
MakeEmpty(T->left);
delete(T);
}
} int main()
{ AvlTree T;
int n, tmp;
cin >> n;
for(int i = ; i < n ; i++)
{
cin >> tmp;
T = Insert(tmp,T);
} return ;
}
AVL树 - 学习笔记的更多相关文章
- zkw线段树学习笔记
zkw线段树学习笔记 今天模拟赛线段树被卡常了,由于我自带常数 \(buff\),所以学了下zkw线段树. 平常的线段树无论是修改还是查询,都是从根开始递归找到区间的,而zkw线段树直接从叶子结点开始 ...
- 仙人掌&圆方树学习笔记
仙人掌&圆方树学习笔记 1.仙人掌 圆方树用来干啥? --处理仙人掌的问题. 仙人掌是啥? (图片来自于\(BZOJ1023\)) --也就是任意一条边只会出现在一个环里面. 当然,如果你的图 ...
- 线段树学习笔记(基础&进阶)(一) | P3372 【模板】线段树 1 题解
什么是线段树 线段树是一棵二叉树,每个结点存储需维护的信息,一般用于处理区间最值.区间和等问题. 线段树的用处 对编号连续的一些点进行修改或者统计操作,修改和统计的复杂度都是 O(log n). 基础 ...
- Treap-平衡树学习笔记
平衡树-Treap学习笔记 最近刚学了Treap 发现这种数据结构真的是--妙啊妙啊~~ 咳咳.... 所以发一发博客,也是为了加深蒟蒻自己的理解 顺便帮助一下各位小伙伴们 切入正题 Treap的结构 ...
- JSOI2008 Blue Mary开公司 | 李超线段树学习笔记
题目链接:戳我 这相当于是一个李超线段树的模板qwqwq,题解就不多说了. 代码如下: #include<iostream> #include<cstdio> #include ...
- Splay伸展树学习笔记
Splay伸展树 有篇Splay入门必看文章 —— CSDN链接 经典引文 空间效率:O(n) 时间效率:O(log n)插入.查找.删除 创造者:Daniel Sleator 和 Robert Ta ...
- CART分类与回归树 学习笔记
CART:Classification and regression tree,分类与回归树.(是二叉树) CART是决策树的一种,主要由特征选择,树的生成和剪枝三部分组成.它主要用来处理分类和回归问 ...
- B和B+树学习笔记
二叉树 如果数据都在内存中,我们就用平衡二叉查找树即可,这样效率最高. 在前面的文章中我使用过红黑树(大致平衡的二叉查找树),500万节点时,搜索的深度可以达到50,也就是需要50次指针操作才能获取到 ...
- Trie树 字典树-学习笔记
字符串--蒟蒻永远的阴影 对于字符串匹配 KMP很好的解决了以一个文本串匹配一个模板串的问题 但如果模板串有多个呢 这是KMP不再适用 我们引入一个新的数据结构--字典树 (当然又有像AC自动机这样更 ...
随机推荐
- new Date()的浏览器兼容性问题
在页面中,我们使用了一个时间上的组件来开发时间选择框,在Chrome下是可以正常运行的,但是发现在IE下是无法正常工作的. 问题出在哪里呢? js从时间获取的时间字符串如果是"-" ...
- svn搭建(linux下)
安装svn: 依赖包: yum install openssl openssl-devel 问题(可以忽略不计):configure: WARNING: we have configured with ...
- python基础-第三篇-函数编程
基本数据类型之set set是无序不允许重复的集合 set创建:s = set() 创建空集合 s = {11,22,33} 转换s = set(可迭代数据) li = [11,22,33,44] ...
- MySQL5.7安装手册
MySQL安装文档 1. 安装依赖包 yum install -y autoconf automake imake libxml2-devel expat-devel cmake gcc gcc-c+ ...
- MYSQL--表分区、查看分区(转)
一. mysql分区简介 数据库分区 数据库分区是一种物理数据库设计技术.虽然分区技术可以实现很多效果,但其主要目的是为了在特定的SQL操作中减少数据读写的总量以缩减sql语句的响应时间, ...
- 【JVM】程序调优
现实企业级Java开发中,有时候我们会碰到下面这些问题: OutOfMemoryError,内存不足 内存泄露 线程死锁 锁争用(Lock Contention) Java进程消耗CPU过高 .... ...
- git学习------>Git 分支管理最佳实践
ps:本文转载于 : https://www.ibm.com/developerworks/cn/java/j-lo-git-mange/index.html Git 是目前最流行的源代码管理工具.大 ...
- 又一次认识java(四) — 组合、聚合与继承的爱恨情仇
有人学了继承,认为他是面向对象特点之中的一个,就在全部能用到继承的地方使用继承,而不考虑到底该不该使用,无疑.这是错误的.那么.到底该怎样使用继承呢? java中类与类之间的关系 大部分的刚開始学习的 ...
- maven工具使用之常用maven命令(二)
1.创建java web项目: # mvn archetype:generate -DgroupId=com.igoodful.sdxs -DartifactId=hubu -Darche ...
- JVM内存—堆(heap)栈(stack)方法区(method) (转)
JAVA的JVM的内存可分为3个区:堆(heap).栈(stack)和方法区(method) 堆区:1.存储的全部是对象,每个对象都包含一个与之对应的class的信息.(class的目的是得到操作指令 ...