文章目录
一、为什么引入平衡二叉树
二、平衡二叉树:(AVL树)
三、平衡二叉树的分析与调整
四、 平衡二叉树的实现
一、为什么引入平衡二叉树
效率高!
对比:二、平衡二叉树:(AVL树)
空树,或者任一结点左、右子树高度差的绝对值不超过1
- 平衡二叉树,也被称为高度平衡树。相比于”二叉查找树”,它的特点是:AVL树中任何节点的两个子树的高度最大差别为1。
- 平衡二叉树的高度是末尾结点到根节点的路径,即边数。高度为0时,总结点数为1.
- 设高度为h的平衡二叉树的最少结点数为n(h)。结点数最少时:
三、平衡二叉树的分析与调整
1.分析
当我们在一个平衡二叉排序树上插入一个结点时,有可能导致失衡,即出现平衡因子绝对值大于一的结点2.平衡调整的四种类型
原则:
- 降低高度
- 保持二叉排序树的性质
1. LL型
- B结点带左子树一起上升
- A结点成为B的右孩子
- 原来B结点的右子树作为A的左子树
2. RR型
- B结点带右子树一起上升
- A结点成为B的左孩子
- 原来B结点的左子树作为A的右子树
图例:
调整后:
3. LR型
- C结点穿过A、B结点上升
- B结点成为C的左孩子
- A结点成为C的右孩子
- 原来C结点的左子树作为B的右子树
- 原来C结点的右子树作为A的左子树
- C原来的左子树变成上升后的左子树的右子树;原来的右子树变成上升后的右子树的左子树
4. RL型
图例:
调整后:
9原来的左子树变成了上升后的左子树的右子树
四、平衡二叉树的实现
1. 平衡二叉树的结构
与二叉树大致相同,多了节点的高度typedef struct BinNode
{
int data;
int height; //当前节点的高度
struct BinNode* lchild;
struct BinNode* rchild;
}BinNode, * BinTree;
//获取当前节点的高度
2. 平衡二叉树的调整
前置函数:int GetHeight(BinTree tree) 。
{
if(tree == NULL)
{
return 0;
}
return tree->height;
}
//判断是否平衡
bool IsBalanced(BinTree tree)
{
int BF = GetHeight(tree->lchild) - GetHeight(tree->rchild);
return abs(BF) < 2;
}
int max(int a,int b)
{
return a > b ? a : b;
}
RR型:
BinTree Rotate_RR(BinTree tree)
{
BinTree temp = tree->rchild;
tree->rchild = temp->lchild;
temp->lchild = tree;
//更新高度,先更新node再更新temp。
tree->height = max(GetHeight(tree->lchild),GetHeight(tree->rchild))+1;
temp->height = max(GetHeight(temp->lchild),GetHeight(temp->rchild))+1;
return temp;
}
LL型:
BinTree Rotate_LL(BinTree tree)
{
BinTree temp = tree->lchild;
tree->lchild = temp->rchild;
temp->rchild = tree;
//更新高度,先更新node再更新temp。
tree->height = max(GetHeight(tree->lchild),GetHeight(tree->rchild))+1;
temp->height = max(GetHeight(temp->lchild),GetHeight(temp->rchild))+1;
return temp;
}
LR型:
BinTree Rotate_LR(BinTree tree)
{
BinTree temp = tree->lchild;
tree->lchild = Rotate_RR(temp);
return Rotate_LL(tree);
}
RL型:
BinTree Rotate_RL(BinTree tree)
{
BinTree temp = tree->rchild;
tree->rchild = Rotate_LL(temp);
return Rotate_RR(tree);
}
3. 平衡二叉树的插入
BinTree Insert(BinTree T, char X)//将X插入到AVL树中,并且返回调整后的AVL树
{
if (!T) //若插入空树,则新建包含一个结点的树
{
T = (BinTree)malloc(sizeof(BinNode));
T->data = X;
T->height = 0;
T->lchild = T->rchild = NULL;
}
else if (X<T->data) //插入T的左子树
{
T->lchild = Insert(T->lchild, X);
if (GetHeight(T->lchild) - GetHeight(T->rchild) == 2) //如果需要右旋
{
if (X<T->lchild->data)
{
T = Rotate_LL(T); //右单旋
}
else
{
T = Rotate_LR(T);//左右双旋
}
}
}
else if (X>T->data) //插入T的左子树
{
T->rchild = Insert(T->rchild, X);
if (GetHeight(T->rchild) - GetHeight(T->lchild) == 2)//如果需要左旋
{
if (X>T->rchild->Data)
{
T = Rotate_RR(T);//左单旋
}
else
{
T = Rotate_RL(T);//左右双旋
}
}
}
T->height = max(GetHeight(T->lchild), GetHeight(T->rchild)) + 1;//更新树高
return T;
}