java实现平衡二叉树

java实现平衡二叉树(AVL树)


给你一个数列{1,2,3,4,5,6},要求创建一颗二叉排序树(BST), 并分析问题所在问题。

左边 BST存在的问题分析:

  • 1)左子树全部为空,从形式上看,更像一个单链表.

  • 2)插入速度没有 影响

  • 3)查询速度明显降低(因为需要依次比较),不能发挥BST树的优势,因为每次还需要比较左子树,其查询速度比单链表还慢

  • 4)解决方案 - 平衡二 叉树(AVL)

基本介绍

平衡二叉树也叫平衡二叉搜索树(Self-balancing binary search tree)又被称为AVL树,可以保证查询效率较高

具有以下特点: 它是一棵空树或它的左右两个子树的高度差的绝对值不超过1,并且左右两个子树都是一棵平衡二叉树。平衡二叉树的常用实现方法有红黑树AVL替罪羊树Treap伸展树等。

举例说明, 看看下面哪些AVL树,为什么?
在这里插入图片描述
平衡二叉搜索树的左右子树,高度差不能大于1,所以前两个满足。

获取数的高度方法

//创建Node节点
class Node {
    int value;
    Node left;
    Node right;

    public Node(int value) {
        this.value = value;
    }

    //返回左子树的高度
    public int leftHeight(){
        if(left == null){
            return 0;
        }
        return left.height();
    }

    //返回右边=子树的高度
    public int rightHeight(){
        if(right == null){
            return 0;
        }
        return right.height();
    }

    //返回当前节点的高度,以该节点为根节点的树的高度
    public int height(){
        //返回当前节点的高度
        return Math.max(
                left == null ? 0 : left.height(),
                right == null ? 0 : right.height()) + 1;
    }
}

单旋转(左旋转)

当二叉排序树加入新的节点时,如果右子树高度 - 左子树高度 > 1 时,需要进行左旋转。

//左旋转的方法 
//调用该方法的是,当前节(理解为根节点)点左子树高度比右子树高度小,进行左旋转
//1、首先创建一个新的节点newNode 并赋给当前节点值
//2、让 newNode.left 指向 根节点的左子树
//3、在让newNode.right 指向 根节点右子树(上移)的左子树
//4、再让根节点value 改为右子树的value
//5、再让根节点(上移右节点)的right 指向 右子树的右子树
//6、让根节点(上移右节点)的left指向newNode节点
private void leftRotate(){
    //创建新的节点,以当前根节点的值 (把当前根节点拷贝作为左子树)
    Node newNode = new Node(value);
    //把新的节点的左子树设置成当前节点的左子树(拷贝的新节点指向根节点的左子树)
    newNode.left = left;
    //把新的节点的,右子树设置成,当前节点右子树的左子树(将根节点右子树的左子树給拷贝节点)
    newNode.right = right.left;
    //把当前节点的值替换成右子节点的值(当前节点值改为右子树的)
    value = right.value;
    //把当前节点的右子树设置成当前节点右子树的右子树(右子树上移作为根节点)
    right = right.right;
    //把当前节点的左子树(左子节点)设置成新的节点(连接下移的新节点【拷贝根节点】)
    left = newNode;
}

单旋转(右旋转)

当二叉排序树加入新的节点时,如果左子树高度 - 右子树高度> 1 时,需要进行右旋转。

//右旋转的方法 
//左旋转2-6方法左右反过来
//该方法在add方法中,每次添加判断当前节点的左右子树高度
private void rightRotate(){
    //创建新的节点,以当前根节点的值 (把当前根节点拷贝作为右子树)
    Node newNode = new Node(value);
    //把新的节点的右子树设置成当前节点的右子树(拷贝的新节点指向根节点的右子树)
    newNode.right = right;
    //把新的节点的,左子树设置成,当前节点左子树的右子树(将根节点左子树的右子树給拷贝节点)
    newNode.left = left.right;
    //把当前节点的值替换成左子节点的值(当前节点值改为左子树的)
    value = left.value;
    //把当前节点的左子树设置成当前节点左子树的左子树(左子树上移作为根节点)
    left = left.left;
    //把当前节点的右子树(右子节点)设置成新的节点(连接下移的新节点【拷贝根节点】)
    right = newNode;
}

双旋转

在这里插入图片描述
在这里插入图片描述

//添加节点的方法
//递归的形式添加节点,需满足二叉排序树的要求
public void add(Node node){
    if(node == null){
        return;
    }

    //判断传入的节点的值,和当前子树的根节点的值关系
    if(node.value < this.value){
        //如果当前节点,左子树为空null
        if(this.left == null){
            this.left = node;
        } else {
            //递归向左子树添加节点
            this.left.add(node);
        }
    } else { //添加节点的值,大于等于当前节点值
        //如果当前节点,右子树为空null
        if(this.right == null){
            this.right = node;
        } else {
            //递归向右子树添加节点
            this.right.add(node);
        }
    }

    //当添加完一个节点后,如果(右子树的高度-左子树的高度) > 1,左旋转
    if(rightHeight() - leftHeight() > 1){
        //如果当前节点的 右子树 的左子树高度 大于它的 右子树高度
        if(right != null && right.leftHeight() > right.rightHeight()){
            //当前节点的(右子树)进行右旋转
            right.rightRotate();
        }
        leftRotate(); // 左旋转..
        return; //必须要加
    }

    //当添加完一个节点后,如果:(左子树的高度-右子树的高度) > 1,右旋转
    if(leftHeight() - rightHeight() > 1){
        //如果当前节点的[左子树]的[[右子树]]高度大于它的[[左子树]]的高度
        if(left != null && left.rightHeight() > left.leftHeight()){
            //先对当前节点的(左子树)->(左旋转)
            left.leftRotate();
        }
        rightRotate(); //右旋转
    }
}

测试案例

public static void main(String[] args) {
    //int[] arr = {4,3,6,5,7,8}; //左旋数组
    //int[] arr = {10,12,8,9,7,6}; //右旋数组
    int[] arr = {10,11,7,6,8,9}; //双旋数组

    //创建一个AVLTree对象
    AVLTree avlTree = new AVLTree();
    //添加节点
    for (int i = 0; i < arr.length; i++) {
        avlTree.add(new Node(arr[i]));
    }

    //遍历
    System.out.println("中序遍历");
    avlTree.infixOrder();

    System.out.println("在没有平衡处理前~~");
    System.out.println("树的高度="+avlTree.getRoot().height());
    System.out.println("树的左子树高度="+avlTree.getRoot().leftHeight());
    System.out.println("树的右子树高度="+avlTree.getRoot().rightHeight());
    System.out.println("当前的根节点="+avlTree.getRoot());
}

测试结果
在这里插入图片描述
链接:完整代码
提取码:1234


版权声明:本文为qq_45731129原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明。