平衡二叉树(AVL)的基本操作(查找、插入、建树)——附带完整代码以及示例

1 定义

  • 概念:在BST(二叉搜索树)的基础上,任意一个结点的左右子树的高度差(平衡因子)的绝对值不超过1。
  • AVL取自于创建它的两位前苏联的数学家(G.M.Adelse-Velskil和E.M.Landis)的名字。

2 基本操作

2.1 存储结构

struct node{
    int data;//结点权值
    int height;//当前子树高度
    node *lchild, *rchild; 
};

2.2 新建结点

//生成一个结点,结点权值为v
node* newNode(int v){
    node* Node = new node;
    Node->data = v;
    Node->height = 1;
    Node->lchild = Node->rchild  = NULL;
    return Node;
}

2.3 获得以root为根结点的子树的当前height

//获得以root为根结点的子树的当前height
int getHeight(node* root){
    if(root == NULL) return 0;
    return root->height;
}

2.4 计算root的平衡因子

//计算root的平衡因子
int getBalandeFactor(node* root){
    return getHeight(root->lchild) - getHeight(root->rchild); 
}

2.5 更新root的height

//更新root的height
void updateHeight(node* root){
    root->height = max(getHeight(root->lchild),  getHeight(root->rchild)) + 1;
}

2.6 插入结点

2.6.1 左单旋转

在这里插入图片描述

//左单旋转(RR平衡旋转),结点A的右孩子的右子树上插入新结点
void L(node* &root){
    node* temp = root->rchild;
    root->rchild = temp->lchild;
    temp->lchild = root;
    updateHeight(root);
    updateHeight(temp);
    root = temp;
}

2.6.2 右单旋转

在这里插入图片描述

//右单旋转(LL平衡旋转),结点A的左孩子的左子树插入新结点
void R(node* &root){
    node* temp = root->lchild;
    root->lchild = temp->rchild;
    temp->rchild = root;
    updateHeight(root);
    updateHeight(temp);
    root = temp;
}

2.6.3 插入结点

/*
BF:平衡因子
树形                            判定条件                             操作        
LL(结点A的左孩子的左子树插入新结点) BF(root) = 2,BF(root->lchild = 1)   右旋       
LR(结点A的左孩子的右子树插入新结点)  BF(root) = 2,BF(root->lchild = -1)  先对root->lchild左旋,再对root右旋
RR(结点A的右孩子的右子树插入新结点) BF(root) = -2,BF(root->lchild = -1)   左旋
RL(结点A的右孩子的左子树插入新结点)  BF(root) = 2,BF(root->lchild = 1)  先对root->lchild右旋,再对root左旋
*/

void Insert(node* &root, int v){
    if(root == NULL){
        root = newNode(v);
        return;
    }

    if(v < root->data){//v比根结点的权值小
        Insert(root->lchild, v);//往左子树插
        updateHeight(root); //更新树高
        if(getBalandeFactor(root) == 2){
            if(getBalandeFactor(root->lchild) == 1){//LL型
                R(root);
            }else if(getBalandeFactor(root->lchild) == -1){//LR型
                L(root->lchild);
                R(root);
            }
        }
    }else{//v比根结点的权值大
        Insert(root->rchild, v);//往右子树插
        updateHeight(root);//更新树高
        if(getBalandeFactor(root) == -2){
            if(getBalandeFactor(root->rchild) == -1){//RR型
                L(root);
            }else if(getBalandeFactor(root->rchild) == 1){//RL型
                R(root->rchild);
                L(root);
            }
        }
    }

2.7 创建AVL树

//AVL树建立
node* Create(int data[], int n){
    node* root = NULL;//新建空根结点
    
    for (int i = 0; i < n; ++i)//将data[0]~data[n-1]插入AVL树中
    {
        Insert(root, data[i]);
    }

    return root;//返回根结点
}

2.8 查找结点值

//查找值
void Search(node* root, int x){
    if(root == NULL){
        printf("Search fail!\n");
        return;
    }

    if(x == root->data){
        printf("%d\n", root->data);
    }else if(x < root->data){
        Search(root->lchild, x);
    }else{
        Search(root->rchild, x);
    }
}

3.0 完整代码

#include <cstdio>
#include <algorithm>

using std::max;

struct node{
    int data, height;
    node *lchild, *rchild;
};

//生成一个结点,结点权值为v
node* newNode(int v){
    node* Node = new node;
    Node->data = v;
    Node->height = 1;
    Node->lchild = Node->rchild  = NULL;
    return Node;
}

//获得以root为根结点的子树的当前height
int getHeight(node* root){
    if(root == NULL) return 0;
    return root->height;
}

//计算root的平衡因子
int getBalandeFactor(node* root){
    return getHeight(root->lchild) - getHeight(root->rchild);
}

//更新root的height
void updateHeight(node* root){
    root->height = max(getHeight(root->lchild),  getHeight(root->rchild)) + 1;
}

//查找值
void Search(node* root, int x){
    if(root == NULL){
        printf("Search fail!\n");
        return;
    }

    if(x == root->data){
        printf("%d\n", root->data);
    }else if(x < root->data){
        Search(root->lchild, x);
    }else{
        Search(root->rchild, x);
    }
}

//左单旋转(RR平衡旋转),结点A的右孩子的右子树上插入新结点
void L(node* &root){
    node* temp = root->rchild;
    root->rchild = temp->lchild;
    temp->lchild = root;
    updateHeight(root);
    updateHeight(temp);
    root = temp;
}

//右单旋转(LL平衡旋转),结点A的左孩子的左子树插入新结点
void R(node* &root){
    node* temp = root->lchild;
    root->lchild = temp->rchild;
    temp->rchild = root;
    updateHeight(root);
    updateHeight(temp);
    root = temp;
}


/*
BF:平衡因子
树形                            判定条件                             操作
LL(结点A的左孩子的左子树插入新结点) BF(root) = 2,BF(root->lchild = 1)   右旋
LR(结点A的左孩子的右子树插入新结点)  BF(root) = 2,BF(root->lchild = -1)  先对root->lchild左旋,再对root右旋
RR(结点A的右孩子的右子树插入新结点) BF(root) = -2,BF(root->lchild = -1)   左旋
RL(结点A的右孩子的左子树插入新结点)  BF(root) = 2,BF(root->lchild = 1)  先对root->lchild右旋,再对root左旋
*/

void Insert(node* &root, int v){
    if(root == NULL){
        root = newNode(v);
        return;
    }

    if(v < root->data){//v比根结点的权值小
        Insert(root->lchild, v);//往左子树插
        updateHeight(root); //更新树高
        if(getBalandeFactor(root) == 2){
            if(getBalandeFactor(root->lchild) == 1){//LL型
                R(root);
            }else if(getBalandeFactor(root->lchild) == -1){//LR型
                L(root->lchild);
                R(root);
            }
        }
    }else{//v比根结点的权值大
        Insert(root->rchild, v);//往右子树插
        updateHeight(root);//更新树高
        if(getBalandeFactor(root) == -2){
            if(getBalandeFactor(root->rchild) == -1){//RR型
                L(root);
            }else if(getBalandeFactor(root->rchild) == 1){//RL型
                R(root->rchild);
                L(root);
            }
        }
    }
}

//AVL树建立
node* Create(int data[], int n){
    node* root = NULL;//新建空根结点

    for (int i = 0; i < n; ++i)//将data[0]~data[n-1]插入AVL树中
    {
        Insert(root, data[i]);
    }

    return root;//返回根结点
}

int main(int argc, char const *argv[])
{
    int tree[7] = {100, 80, 60, 90, 120, 130, 110};
    node* root = Create(tree, 7);

    Search(root, 110);
    Search(root,1);
    return 0;
}
发布了321 篇原创文章 · 获赞 51 · 访问量 4万+

猜你喜欢

转载自blog.csdn.net/qq_33375598/article/details/104260424
今日推荐