简介
链表较低的查找效率推动了二叉搜索树的诞生。二叉搜索树可以以对数级别的时间复杂度进行查找。
然而二叉搜索树在某些特定情况时(数据有序插入)会退化成类似于链表的结构,导致其搜索时的时间复杂度变为O(n)。
AVL树可以解决二叉树搜索树退化的问题。 它通过判断二叉树是否失衡,并在失衡时通过旋转将失衡的二叉树置为平衡状态,以防止其变为链表。
原理
AVL树的旋转方式有两种 —— 左旋和右旋
左旋使失衡节点(左右子树高度差的绝对值大于1的节点)围绕将成为新根的节点向左(逆时针)旋转,成为新根节点的左孩子,如果新根节点已经有了左孩子,则将其插到失衡节点的右边。
右旋使失衡节点(左右子树高度差的绝对值大于1的节点)围绕将成为新根的节点向右(顺时针)旋转,成为新根节点的右孩子,如果新根节点已经有了右孩子,则将其插到失衡节点的左边。
AVL树引入平衡因子的概念描述树的平衡状态。平衡因子是左边的节点高度减右边的节点高度。
AVL树有四种失衡状态
- LL型 失衡节点的平衡因子是2 其左孩子的平衡因子是1 需要右旋
- RR型 失衡节点的平衡因子是-2 其右孩子的平衡因子是-1 需要左旋
- LR型 失衡节点的平衡因子是2 其左孩子的平衡因子是-1 需要先左旋左孩子再右旋失衡节点
- RL型 失衡节点的平衡因子是-2 其右孩子的平衡因子是1 需要先右旋右孩子再左旋失衡节点
实现
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
| struct Node { int val; int height; Node * left; Node * right; Node(int); };
Node::Node(int val) { this->height = 1; this->left = nullptr; this->right = nullptr; this->val = val; }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| int getHeight(Node * target) { if(target == nullptr) { return 0; } return target->height; }
void updateHeight(Node * target) { target->height = max(getHeight(target->left),getHeight(target->right)) + 1; }
int getBalance(Node * target) { return getHeight(target->left) - getHeight(target->right); }
|
遍历
1 2 3 4 5 6 7 8 9 10
| void InOrder(Node * root) { if(root == nullptr) { return ; }
InOrder(root->left); cout<<root->val<<endl; InOrder(root->right); }
|
旋转
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
| void LeftRotate(Node ** root) { Node * new_root = (*root)->right; Node * T = new_root->left;
new_root->left = *root; (*root)->right = T;
updateHeight(*root); updateHeight(new_root);
*root = new_root; }
void RightRotate(Node ** root) { Node * new_root = (*root)->left; Node * T = new_root->right;
new_root->right = (*root); (*root)->left = T;
updateHeight(*root); updateHeight(new_root);
*root = new_root; }
|
插入
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
| void Insert(Node ** root,int val) { if(*root == nullptr) { *root = new Node(val); return ; }
if(val == (*root)->val) { return; }
if(val<(*root)->val) { Insert(&(*root)->left,val); } else if(val>(*root)->val) { Insert(&(*root)->right,val); } updateHeight(*root);
int balance = getBalance(*root);
if(balance == 2) { if(getBalance((*root)->left) == 1) { RightRotate(root); } else if(getBalance((*root)->left) == -1) { LeftRotate(&(*root)->left); RightRotate(&(*root)); } } else if(balance == -2) { if(getBalance((*root)->right) == 1) { RightRotate(&(*root)->right); LeftRotate(root); } else if(getBalance((*root)->right) == -1) { LeftRotate(root); } } }
|
删除
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
| Node * findLeftMax(Node * root) { Node * max = root->left; while (max && max->right) { max = max->right; } return max; }
void remove(Node ** root,int val) { if(*root == nullptr) { return; }
if(val<(*root)->val) { remove(&(*root)->left,val); } else if(val > (*root)->val) { remove(&(*root)->right,val); } else { Node * temp = *root; if(!(*root)->left && !(*root)->right) { delete *root; *root = nullptr; } else if((*root)->left && !(*root)->right) { *root = (*root)->left; delete temp; } else if((*root)->right && !(*root)->left) { *root = (*root)->right; delete temp; } else { Node * leftMax = findLeftMax(*root); (*root)->val = leftMax->val; remove(&(*root)->left,leftMax->val); } } if(*root == nullptr ) { return; }
updateHeight(*root);
int balance = getBalance(*root);
if(balance == 2) { int lb = getBalance((*root)->left); if(lb == 1) { RightRotate(root); } else if(lb == -1) { LeftRotate(&(*root)->left); RightRotate(root); } } else if(balance == -2) { int rb = getBalance((*root)->right); if(rb == -1) { LeftRotate(root); } else if(rb == 1) { RightRotate(&(*root)->right); LeftRotate(root); } } }
|
完整代码