广告
返回顶部
首页 > 资讯 > 精选 >Java数据结构之AVL树实例分析
  • 911
分享到

Java数据结构之AVL树实例分析

2023-06-30 18:06:40 911人浏览 八月长安
摘要

这篇文章主要介绍“Java数据结构之AVL树实例分析”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Java数据结构之AVL树实例分析”文章能帮助大家解决问题。AVL树的引入搜索二叉树有着极高的搜索效

这篇文章主要介绍“Java数据结构之AVL树实例分析”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Java数据结构之AVL树实例分析”文章能帮助大家解决问题。

Java数据结构之AVL树实例分析

AVL树的引入

搜索二叉树有着极高的搜索效率,但是搜索二叉树会出现以下极端情况:
Java数据结构之AVL树实例分析
这样的二叉树搜索效率甚至比链表还低。在搜索二叉树基础上出现的平衡二叉树(AVL树)就解决了这样的问题。当平衡二叉树(AVL树)的某个节点左右子树高度差的绝对值大于1时,就会通过旋转操作减小它们的高度差。

基本概念

AVL树本质上还是一棵二叉搜索树,它的特点是:

  1. 本身首先是一棵二叉搜索树

  2. 每个结点的左右子树的高度之差的绝对值(平衡因子)最多为1。也就是说,AVL树,本质上是带了平衡功能的二叉查找树(二叉排序树,二叉搜索树)。

  3. 当插入一个节点或者删除一个节点时,导致某一个节点的左右子树高度差的绝对值大于1,这时需要通过左旋右旋的操作使二叉树再次达到平衡状态。

平衡因子(balanceFactor)

  • 一个结点的左子树与右子树的高度之差

  • AVL树中的任意结点的BF只可能是-1,0和1。

基础设计

下面是AVL树需要的简单方法和属性:

public class AVLTree <E extends Comparable<E>>{    class node{        E value;        Node left;        Node right;        int height;        public Node(){}        public Node(E value){            this.value = value;            height = 1;            left = null;            right = null;        }        public void display(){            System.out.print(this.value + " ");        }    }    Node root;    int size;    public int size(){        return size;    }    public int getHeight(Node node) {        if(node == null) return 0;        return node.height;    }    //获取平衡因子(左右子树的高度差,大小为1或者0是平衡的,大小大于1不平衡)    public int getBalanceFactor(){        return getBalanceFactor(root);    }    public int getBalanceFactor(Node node){        if(node == null) return 0;        return getHeight(node.left) - getHeight(node.right);    }    //判断一个树是否是一个平衡二叉树    public boolean isBalance(Node node){        if(node == null) return true;        int balanceFactor = Math.abs(getBalanceFactor(node.left) - getBalanceFactor(node.right));        if(balanceFactor > 1) return false;        return isBalance(node.left) && isBalance(node.right);    }    public boolean isBalance(){        return isBalance(root);    }    //中序遍历树    private  void inPrevOrder(Node root){        if(root == null) return;        inPrevOrder(root.left);        root.display();        inPrevOrder(root.right);    }    public void inPrevOrder(){        System.out.print("中序遍历:");        inPrevOrder(root);    }}

RR(左旋)

往一个树右子树的右子树上插入一个节点,导致二叉树变得不在平衡,如下图,往平衡二叉树中插入5,导致这个树变得不再平衡,此时需要左旋操作,如下:
Java数据结构之AVL树实例分析
代码如下:

//左旋,并且返回新的根节点    public Node leftRotate(Node node){        System.out.println("leftRotate");       Node cur = node.right;       node.right = cur.left;       cur.left = node;       //跟新node和cur的高度        node.height = Math.max(getHeight(node.left),getHeight(node.right)) + 1;        cur.height = Math.max(getHeight(cur.left),getHeight(cur.right)) + 1;        return cur;    }

LL(右旋)

往一个AVL树左子树的左子树上插入一个节点,导致二叉树变得不在平衡,如下图,往平衡二叉树中插入2,导致这个树变得不再平衡,此时需要左旋操作,如下:
Java数据结构之AVL树实例分析
代码如下:

 //右旋,并且返回新的根节点    public Node rightRotate(Node node){        System.out.println("rightRotate");        Node cur = node.left;        node.left = cur.right;        cur.right = node;        //跟新node和cur的高度        node.height = Math.max(getHeight(node.left),getHeight(node.right)) + 1;        cur.height = Math.max(getHeight(cur.left),getHeight(cur.right)) + 1;        return cur;    }

LR(先左旋再右旋)

往AVL树左子树的右子树上插入一个节点,导致该树不再平衡,需要先对左子树进行左旋,再对整棵树右旋,如下图所示,插入节点为5.
Java数据结构之AVL树实例分析

RL(先右旋再左旋)

往AVL树右子树的左子树上插入一个节点,导致该树不再平衡,需要先对右子树进行右旋,再对整棵树左旋,如下图所示,插入节点为2.
Java数据结构之AVL树实例分析

添加节点

//添加元素    public  void add(E e){        root = add(root,e);    }    public Node add(Node node, E value) {        if (node == null) {            size++;            return new Node(value);        }        if (value.compareTo(node.value) > 0) {            node.right = add(node.right, value);        } else if (value.compareTo(node.value) < 0) {            node.left = add(node.left, value);        }        //跟新节点高度        node.height = Math.max(getHeight(node.left), getHeight(node.right)) + 1;        //获取当前节点的平衡因子        int balanceFactor = getBalanceFactor(node);        //该子树不平衡且新插入节点(导致不平衡的节点)在左子树的左子树上,此时需要进行右旋        if (balanceFactor > 1 && getBalanceFactor(node.left) >= 0) {            return rightRotate(node);        }        //该子树不平衡且新插入节点(导致不平衡的节点)在右子树子树的右子树上,此时需要进行左旋        else if (balanceFactor < -1 && getBalanceFactor(node.right) <= 0) {            return leftRotate(node);        }        //该子树不平衡且新插入节点(导致不平衡的节点)在左子树的右子树上,此时需要先对左子树左旋,在整个树右旋        else if (balanceFactor > 1 && getBalanceFactor(node.left) < 0) {            node.left = leftRotate(node.left);            return rightRotate(node);        }        //balanceFactor < -1 && getBalanceFactor(node.left) > 0        //该子树不平衡且新插入节点(导致不平衡的节点)在右子树的左子树上,此时需要先对右子树右旋,再整个树左旋        else if(balanceFactor < -1 && getBalanceFactor(node.right) > 0) {            node.right = rightRotate(node.right);            return leftRotate(node);        }        return node;    }

删除节点

 //删除节点    public E remove(E value){        root = remove(root,value);        if(root == null){            return null;        }        return root.value;    }    public Node remove(Node node, E value){        Node retNode = null;        if(node == null)            return retNode;        if(value.compareTo(node.value) > 0){            node.right = remove(node.right,value);            retNode = node;        }        else if(value.compareTo(node.value) < 0){            node.left = remove(node.left,value);            retNode = node;        }        //value.compareTo(node.value) = 0        else{            //左右节点都为空,或者左节点为空            if(node.left == null){                size--;                retNode = node.right;            }            //右节点为空            else if(node.right == null){                size--;                retNode = node.left;            }            //左右节点都不为空            else{                Node successor = new Node();                //寻找右子树最小的节点                Node cur = node.right;                while(cur.left != null){                    cur = cur.left;                }                successor.value  = cur.value;                successor.right = remove(node.right,value);                successor.left = node.left;                node.left =  node.right = null;                retNode = successor;            }            if(retNode == null)                return null;            //维护二叉树平衡            //跟新height            retNode.height = Math.max(getHeight(retNode.left),getHeight(retNode.right));        }        int balanceFactor = getBalanceFactor(retNode);        //该子树不平衡且新插入节点(导致不平衡的节点)在左子树的左子树上,此时需要进行右旋        if (balanceFactor > 1 && getBalanceFactor(retNode.left) >= 0) {            return rightRotate(retNode);        }        //该子树不平衡且新插入节点(导致不平衡的节点)在右子树子树的右子树上,此时需要进行左旋        else if (balanceFactor < -1 && getBalanceFactor(retNode.right) <= 0) {            return leftRotate(retNode);        }        //该子树不平衡且新插入节点(导致不平衡的节点)在左子树的右子树上,此时需要先对左子树左旋,在整个树右旋        else if (balanceFactor > 1 && getBalanceFactor(retNode.left) < 0) {            retNode.left = leftRotate(retNode.left);            return rightRotate(retNode);        }        //该子树不平衡且新插入节点(导致不平衡的节点)在右子树的左子树上,此时需要先对右子树右旋,再整个树左旋        else if(balanceFactor < -1 && getBalanceFactor(retNode.right) > 0) {            retNode.right = rightRotate(retNode.right);            return leftRotate(retNode);        }        return  retNode;    }

关于“Java数据结构之AVL树实例分析”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识,可以关注编程网精选频道,小编每天都会为大家更新不同的知识点。

--结束END--

本文标题: Java数据结构之AVL树实例分析

本文链接: https://www.lsjlt.com/news/330620.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Java数据结构之AVL树实例分析
    这篇文章主要介绍“Java数据结构之AVL树实例分析”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“Java数据结构之AVL树实例分析”文章能帮助大家解决问题。AVL树的引入搜索二叉树有着极高的搜索效...
    99+
    2023-06-30
  • C++数据结构AVL树全面分析
    目录概念AVL树的实现AVL树的节点定义AVL树的插入方法概述平衡因子的调节插入代码实现AVL树的查找AVL树的删除方法概述平衡因子的调节删除代码如下AVL树的测试代码总结⭐️博客代...
    99+
    2022-11-12
  • C++数据结构之AVL树的实现
    目录1.概念(1)二叉搜索树的缺点(2)定义节点2.插入(1)拆分(2)找节点与插节点(3)更新平衡因子与旋转3.判断4.完整代码及测试代码完整代码测试代码1.概念 (1)二叉搜索树...
    99+
    2022-11-13
  • java数据结构之树的示例分析
    这篇文章主要介绍java数据结构之树的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!树定义和基本术语定义树(Tree)是n(n≥0)个结点的有限集T,并且当n>0时满足下列条件:(1)有且仅有一个特定的称为根...
    99+
    2023-05-30
    java
  • C++数据结构之AVL树如何实现
    这篇文章主要讲解了“C++数据结构之AVL树如何实现”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“C++数据结构之AVL树如何实现”吧!1.概念(1)二叉搜索树的缺点要手撕AVL树,我们首先...
    99+
    2023-07-02
  • Java数据结构之二叉搜索树实例分析
    这篇文章主要介绍“Java数据结构之二叉搜索树实例分析”,在日常操作中,相信很多人在Java数据结构之二叉搜索树实例分析问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java数据结构之二叉搜索树实例分析”的疑...
    99+
    2023-06-30
  • Java数据结构之红黑树的示例分析
    小编给大家分享一下Java数据结构之红黑树的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!一、红黑树所处数据结构的位置:在JDK源码中, 有treeMap...
    99+
    2023-05-30
    java
  • Java数据结构之实现哈夫曼树的示例分析
    这篇文章主要介绍了Java数据结构之实现哈夫曼树的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。一、与哈夫曼树相关的概念概念含义1. 路径从树中一个结点到另一个结点的...
    99+
    2023-06-15
  • JavaScript之树结构的示例分析
    这篇文章主要介绍了JavaScript之树结构的示例分析,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。二叉树--概念--二叉树是一种树形结构...
    99+
    2022-10-19
  • 图解AVL树数据结构输入与输出及实现示例
    目录AVL树(平衡二叉树):AVL树的作用:AVL树的基本操作:AVL树的插入,单旋转的第一种情况---右旋:AVL树的插入,单旋转的第二种情况---左旋:AVL树的插入,双旋转的第...
    99+
    2022-11-13
  • AVL树数据结构输入与输出怎么实现
    本篇内容介绍了“AVL树数据结构输入与输出怎么实现”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!AVL树(平衡二叉树):AVL树本质上是一颗...
    99+
    2023-06-30
  • Java数据结构之优先级队列实例分析
    本文小编为大家详细介绍“Java数据结构之优先级队列实例分析”,内容详细,步骤清晰,细节处理妥当,希望这篇“Java数据结构之优先级队列实例分析”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。一、堆的概念堆的定义:...
    99+
    2023-06-29
  • Java数据结构之链表的示例分析
    小编给大家分享一下Java数据结构之链表的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!一、链表的介绍什么是链表链表是一种物理存储单元上非连续、非顺序的存...
    99+
    2023-06-15
  • Java数据结构学习之树
    目录一、树1.1 概念1.2 术语1.3 树的实现1.3.1 用数组来实现一棵树?1.3.2 用链表实现一棵树?1.3.3 树的转化1.4 二叉树1.4.1 二叉树的性质1.4.2 ...
    99+
    2022-11-12
  • C++数据结构红黑树的示例分析
    这篇文章给大家分享的是有关C++数据结构红黑树的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。概念和性质红黑树的概念: 红黑树,是一种二叉搜索树,但在每个结点上增加一个存储位表示结点的颜色,可以是Red或...
    99+
    2023-06-29
  • javascript数据结构之多叉树经典操作的示例分析
    这篇文章给大家分享的是有关javascript数据结构之多叉树经典操作的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。多叉树可以实现复杂的数据结构的存储,通过遍历方法可以...
    99+
    2022-10-19
  • JavaScript数据结构与算法之栈实例分析
    这篇文章主要介绍了JavaScript数据结构与算法之栈实例分析的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇JavaScript数据结构与算法之栈实例分析文章都会有所收获,下面我们一起来看看吧。1.认识栈栈:...
    99+
    2023-07-02
  • PHP数据结构之图存储结构的示例分析
    这篇文章主要介绍PHP数据结构之图存储结构的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!图的存储结构图的概念介绍得差不多了,大家可以消化消化再继续学习后面的内容。如果没有什么问题的话,我们就继续学习接下来的...
    99+
    2023-06-20
  • Java数据结构学习之二叉树
    一、背景知识:树(Tree) 在之前的笔记中,我们介绍的链表、栈、队列、数组和字符串都是以线性结构来组织数据的。本篇笔记要介绍的树采用的是树状结构,这是一种非线性的数据组织形式。 树...
    99+
    2022-11-12
  • java数据结构之搜索二叉树
    本文实例为大家分享了java数据结构之搜索二叉树的具体代码,供大家参考,具体内容如下 搜索二叉树的定义是:在一个二叉树上,左节点一定比父节点小,右节点一定比父节点大,其他定义跟二叉树...
    99+
    2022-11-12
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作