数据结构与算法(十六):平衡二叉树

数据结构与算法(十六):平衡二叉树一、什么是平衡二叉树1.概述平衡二叉树(AVL树)是一种带有平衡条件的二叉搜索树。它的特性如下:AVL树的左右两个子树的高度差的绝对值不超过1AVL树的左右两个子树都是一棵平衡二叉树举个例子

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

一、什么是平衡二叉树

1.概述

平衡二叉树(AVL树)是一种带有平衡条件的二叉搜索树。它的特性如下:

  • AVL树的左右两个子树的高度差的绝对值不超过1
  • AVL树的左右两个子树都是一棵平衡二叉树

image-20200722173142958

举个例子,如上图所示:

  • 第一棵树左树高2,右树高1,差值为1,是一颗AVL树;
  • 第二棵树左树高2,右树高2,差值为0,是一颗AVL树;
  • 第三棵树左树高3,右树高1,差值为2,不是一颗AVL树;

红黑树就是一直AVL树。

2.为什么需要平衡二叉树

当我们使用二叉排序树的时候,当连续插入顺序的节点的时候就会出现问题。比如,我们插入{1,2,3,4,5}这样一个数组:

image-20200722173840094

可见该树左树节点全为空,比起树更像单链表,这也导致了该树的插入和查询速度明显的下降,查询速度甚至因为每次多处一个比较左树的操作导致还不如单链表。为了避免这种情况,我们引入的AVL树。

二、AVL树左旋转

1.思路分析

AVL为了避免左右树高度差超过1,在可能导致这种情况的插入或者删除操作时会进行旋转。

我们举个例子,现在有数列{4,3,6,5,7},当插入8后,现在的得到的排序树如下图:

image-20200723175058126

明显不再是一个AVL树,所以需要进行左旋转

  1. 我们以当前根节点值再创建一个新节点newNode

  2. 让新节点的左子节点指向根节点的左子节点

    newNode.left = root.left

  3. 让新节点的右子节点指向根节点的右子节点的左子节点

    newNode.right = root.right.left

    image-20200723183026891

  4. 把根节点的值换成右子节点的值

    root.val = root.right.val

  5. 把根节点的右子节点指向其右子节点的右子节点

    root.right = root.right.right

  6. 让根节点的左子节点指向新节点(根节点的右子节点成为了新的根节点)

    root.left = newNode

    image-20200723183950837

我们调整一下图片样式,就可以直观的看到左旋转后树的样子:

image-20200723184527438

网上看到一个非常形象直观的动图:

数据结构与算法(十六):平衡二叉树

不难理解:左旋的目的是降低左子树的高度

2.代码实现

由于AVL树是基于BST改进的一种数据结构,所以这里的AVL树类继承了BST的方法和代码,使用同一个节点类,这里具体的代码可以参考之前的文章

我们先创建一个继承BST的AVL树类:

/**
 * @Author:CreateSequence
 * @Date:2020-07-23 19:01
 * @Description:平衡二叉树
 * 由于是在二叉排序树的基础上改进,这里直接继承了二叉排序树类
 */
public class AVLTree extends BinarySortTree{

    public AVLTree(BinarySortTreeNode root) {
        super(root);
    }
    
}

由于旋转的条件是左右子树高度差大于1,所以我们需要有几个方法来判断树的高度:

/**
 * 获取当前节点的右子树高度
 * @param node
 * @return
 */
public int getRightHeight(BinarySortTreeNode node) {
    if (node.right == null) {
        return 0;
    }
    return getHeight(node.right);
}

/**
 * 获取当前节点的左子树高度
 * @param node
 * @return
 */
public int getLeftHeight(BinarySortTreeNode node){
    if (node.left == null) {
        return 0;
    }
    return getHeight(node.left);
}

/**
 * 获取以当前节点为根节点的树高度
 * @param node
 * @return
 */
public int getHeight(BinarySortTreeNode node) {
    //判断当前节点的左/右节点是否为空,是返回0,否则遍历返回当前节点的左右树最高值
    return Math.max(node.left == null ? 0 : getHeight(node.left), node.right == null ? 0 : getHeight(node.right)) + 1;
}

接着我们需要一个让树左旋的代码,步骤同思路分析:

/**
 * 排序树左旋转
 */
private void leftRotate() {
    // 1.创建新节点,与根节点值相同
    BinarySortTreeNode node = new BinarySortTreeNode(root.val);
    //2.让新节点左子节点指向根节点左子节点
    node.left = root.left;
    //3.让新节点的右子节点指向根节点的右子节点的左子节点
    node.right = root.right.left;
    //4.让根节点的值变为其右子节点的值
    root.val = root.right.val;
    //5.把根节点的右子节点指向其右子节点的右子节点
    root.right = root.right.right;
    //6.让根节点的左子节点指向新节点
    root.left = node;
}

然后我们再原先旧的添加方法上进行改进:

当添加完一个节点后,我们判断左右子树的高度差是否大于1,如果是就进行左旋

/**
 * 重写二叉排序树的节点添加方法,当添加完节点后左子树与右子树高度差大于1时,让树进行左旋转,若情况相反则进行右旋转
 * @param node
 */
@Override
public void add(BinarySortTreeNode node) {
    super.add(node);
    //添加完节点后,判断左子树与右子树高度差是否大于1
    int disparity = getRightHeight(root) - getLeftHeight(root);
    if (disparity > 1) {
        System.out.println("高度差:" + disparity + ",左旋转!");
        //左子树与右子树高度差大于1就左旋
        leftRotate();
    }
}

注意:截止目前,仅仅只对左子树高度较高的情况作了处理!

三、AVL树的双旋转

左旋转是为了降低左子树的高度,但是如果是右子树高度过高,我们就需要右旋,事实上,一个完整的AVL树,应当是能够双旋的。

右旋的步骤与左旋基本一致,但是方向不同:

  1. 我们以当前根节点值再创建一个新节点newNode

  2. 让新节点的右子节点指向根节点的右子节点

    newNode.right = root.right

  3. 让新节点的左子节点指向根节点的左子节点右子节点

    newNode.left = root.left.right

  4. 把根节点的值换成左子节点的值

    root.val = root.left.val

  5. 把根节点的左子节点指向其左子节点左子节点

    root.left = root.left.left

  6. 让根节点的右子节点指向新节点(根节点的左子节点成为了新的根节点)

    root.right = newNode

数据结构与算法(十六):平衡二叉树

实现代码:

/**
 * 排序树右旋转
 */
private void rightRotate() {
    // 1.创建新节点,与根节点值相同
    BinarySortTreeNode node = new BinarySortTreeNode(root.val);
    //2.让新节点右子节点指向根节点右子节点
    node.right = root.right;
    //3.让新节点的左子节点指向根节点的左子节点的右子节点
    node.left = root.left.right;
    //4.让根节点的值变为其左子节点的值
    root.val = root.left.val;
    //5.把根节点的左子节点指向其左子节点的左子节点
    root.left = root.left.left;
    //6.让根节点的右子节点指向新节点
    root.right = node;
}

现在为排序树的add方法添加右旋的情况:

/**
 * 重写二叉排序树的节点添加方法,当添加完节点后左子树与右子树高度差大于1时,让树进行左旋转,若情况相反则进行右旋转
 * @param node
 */
@Override
public void add(BinarySortTreeNode node) {
    super.add(node);
    //添加完节点后,判断左右树高度差是否大于1
    int disparity = getRightHeight(root) - getLeftHeight(root);
    if (disparity > 1) {
        System.out.println("高度差:" + disparity + ",左旋转!");
        //左子树与右子树高度差大于1就左旋
        leftRotate();
    }else if (- disparity > 1){
        //右子树与左子树高度差小于1就左旋
        rightRotate();
    }
}
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/170801.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 深入浅出Python——Python基础语法全解

    深入浅出Python——Python基础语法全解前言:Python是一个高层次的结合了解释性、编译性、互动性和面向对象的脚本语言。文章目录一、Python简介1.了解Python2.Python介绍3.Python特点4.Python发展历史5.Python版本二、Python解释器1.解释器的作用2.解释器的安装三、PyCharm安装与使用1.PyCharm的作用2.PyCharm安装与使用四、注释1.注释的作用2.注释的分类及语法五、变量1.变量的作用2.定义变量2.1标识符2.2命名习惯2.3使用变量2.4认识

    2022年6月24日
    21
  • python编程前景_Python前景如何,学完后可以从事方向?

    python编程前景_Python前景如何,学完后可以从事方向?前段时间浙江八年级新增了Python编程的课程,消息一出,引起了很多人的关注。连中学生都在学Python了,你还在犹豫要不要学习吗?对于想学Python,却又担心Python前景以及学完后可以从事方向的人,下面,小雷就给大家介绍一下。Python前景怎么样?目前国内外很多公司都在使用Python,例如搜索引擎Google的核心代码是Python完成的、迪士尼公司动画生成的Unix版本都内建了Pyt…

    2022年5月16日
    40
  • 设置ntp服务器同步时间_安卓设置ntp服务器地址

    设置ntp服务器同步时间_安卓设置ntp服务器地址有时服务器需要调整时区并调整时间,需要用到的命令:ntpdate一般Linux系统都默认安装了NTP服务,如果没有安装的话,也可以直接使用yum安装,yum安装命令为:yuminstall-yntpdate首先修改一下时区为上海时区:cp/usr/share/zoneinfo/Asia/Shanghai/etc/localtime然后选择国家授时中心的服务器地址:ntpdate210…

    2022年5月3日
    309
  • STemWin学习:关于窗口消息的基础知识

    STemWin学习:关于窗口消息的基础知识刚开始接触emWin,记录一下我自己感悟的心得。首先从GUIBuilder小工具创建的窗口文件讲解。//USERSTART(Optionallyinsertadditionalstaticdata)#defineBUTTON_SIZE_X20#defineBUTTON_SIZE_Y20#defineBUTTON_START_X55#define…

    2025年6月29日
    2
  • 固态硬盘开盘数据恢复的方法是_硬盘数据恢复原理

    固态硬盘开盘数据恢复的方法是_硬盘数据恢复原理在电脑的使用中有时因为一些不当的操作会导致固态硬盘损坏,有的网友就在现实中遇到了这种情况,咨询小编固态硬盘开盘数据恢复的方法,下面小编就将怎么恢复固态硬盘数据教给大家。更多一键重装系统的方法在这里工具/原料系统版本:win10教育版品牌型号:华为MateBookXPro方法一、固态硬盘开盘数据恢复的方法1、怎么恢复固态硬盘数据呢,首先可以查看回收站,如果被删除的数据还在回收站里点击还原即可。方法二、固态硬盘开盘数据恢复的方法1、下载安装嗨格式数据恢复大师,在首界面选择恢复模式和文件存储位置,点击扫描,

    2022年9月20日
    3
  • 阿里云之自动化构建方案

    阿里云之自动化构建方案

    2020年11月19日
    199

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号