动态查找-AVL树

二叉平衡树(Balanced Binary Tree 或 Height-Balanced Tree)(AVL)

定义:它是一棵空树,或者是具有下列性质的二叉树:
它的左子树和右子树都是平衡二叉树,且左子树和右子树的深度之差的绝对值不超过1.
二叉树上的节点的平衡因子BF(Balance Factor)定义为该结点的左子树的深度减去它的右子树的深度,则平衡二叉树上所有结点的平衡因子只可能是-1、0和1。

AVL树的实现:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
#include <stdio.h>
#include <stdlib.h>

#define LH +1//左边高
#define EH 0//一样高
#define RH -1//右边高
typedef struct BSTNode{
int data;
int bf;//平衡因子
struct BSTNode *lchild, *rchild;
}BSTNode, *BSTree;//平衡二叉排序树结构的定义
void R_Rotate(BSTree &p)
{//对以*p为根的二叉排序树做右旋处理,处理之后p指向新的树根结点

BSTree lc;
lc = p->lchild;
p->lchild = lc->rchild;
lc->rchild = p; p = lc;
}
void L_Rotate(BSTree &p)
{//对以*p为根的二叉排序树做左旋处理,处理之后p指向新的树根结点

BSTree rc;
rc = p->rchild;
p->rchild = rc->lchild;
rc->lchild = p; p = rc;
}
void LeftBalance(BSTree &T)
{//对已*T为根的二叉排序树做左平衡旋转处理

BSTree lc, rd;
lc = T->lchild;
switch (lc->bf)
{
case LH:T->bf = lc->bf = EH;
R_Rotate(T);
break;
case RH:rd = lc->rchild;
switch (rd->bf)
{
case LH:T->bf = RH; lc->bf = EH;
break;
case EH:T->bf = lc->bf = EH;
break;
case RH:T->bf = EH; lc->bf = RH;
break;
}
rd->bf = EH;
L_Rotate(T->lchild);
R_Rotate(T);
}
}
void RightBalance(BSTree &T)
{//对已*T为根的二叉排序树做右平衡旋转处理

BSTree lc, rd;
lc = T->rchild;
switch (lc->bf)
{
case RH:T->bf = lc->bf = EH;
L_Rotate(T);
break;
case LH:rd = lc->lchild;
switch (rd->bf)
{
case RH:T->bf = LH; lc->bf = EH;
break;
case LH:T->bf = EH; lc->bf = RH;
break;
case EH:T->bf = lc->bf = EH;
break;
}
rd->bf = EH;
R_Rotate(T->rchild);
L_Rotate(T);
}
}
int InsertAVL(BSTree &T, int key, bool &taller)
{//若在平衡二叉排序树中不存在与关键字key相等的结点,则将关键字插入树中

//布尔变量taller表示树是否“长高”
if (T == NULL)
{
T = (BSTree)malloc(sizeof(BSTNode));
T->data = key; T->bf = EH;//叶子结点其平衡因子肯定为0
T->lchild = T->rchild = NULL;
taller = 1;//树长高了
}
else
{
if (key == T->data)
{//如果树中已存放此关键字则不予插入
taller = 0;
return 0;
}
if (key<T->data)
{//关键字小于根结点则插入其左子树中
if (!InsertAVL(T->lchild, key, taller))
return 0;
if (taller)
{//如果树长高了,判断是否平衡
switch (T->bf)
{
case LH:LeftBalance(T);//不平衡时调用左平衡函数,使左子树平衡
taller = 0; break;
case EH:T->bf = LH; taller = 1;
break;
case RH:T->bf = EH; taller = 0;
break;
}
}
}
else
{//插入右子树中
if (!InsertAVL(T->rchild, key, taller))
return 0;
if (taller)
{
switch (T->bf)
{
case LH:T->bf = EH; taller = 0;
break;
case EH:T->bf = RH; taller = 1;
break;
case RH:RightBalance(T); taller = 0;
break;
}
}
}
}
return 1;
}

void createAVL(BSTree &T){
int a[10] = { 4, 7, 2, 1, 5, 9, 8, 0, 3, 6 };
bool taller = false;
T = NULL;
for (int i = 0; i < 10; i++){
InsertAVL(T, a[i],taller);
}
}

void InOrderAVL(BSTree T){
if (T){
InOrderAVL(T->lchild);
printf("%d ", T->data);
InOrderAVL(T->rchild);
}
}

int main(){
BSTree T;
createAVL(T);
InOrderAVL(T);
return 0;
}

文章目录
  1. 1. 二叉平衡树(Balanced Binary Tree 或 Height-Balanced Tree)(AVL)
,