首页 > 编程语言 >Java实现平衡二叉搜索树(AVL树)

Java实现平衡二叉搜索树(AVL树)

时间:2022-11-23 19:45:35浏览次数:40  
标签:Node right Java 二叉 AVL new root public left

上一篇实现了二叉搜索树,本章对二叉搜索树进行改造使之成为平衡二叉搜索树(Balanced Binary Search Tree)。

不平衡的二叉搜索树在极端情况下很容易退变成链表,与新增/删除/查找时间复杂度为O(logN)的目标又远了一步。

平衡二叉搜索树始终围绕O(logN)这个目标来构建数据结构。

节点类和实现类:

/**
 * 二叉搜索树节点&平衡二叉搜索树添加节点时自平衡实现
 */
public class Node
{
    private int data;//数据域
    private Node left;//左节点(左孩子)
    private Node right;//右节点(右孩子)

    public int getData() {
        return data;
    }
    public void setData(int data) {
        this.data = data;
    }
    public Node getLeft() {
        return left;
    }
    public void setLeft(Node left) {
        this.left = left;
    }
    public Node getRight() {
        return right;
    }
    public void setRight(Node right) {
        this.right = right;
    }

    //构造函数
    public Node(int data, Node left, Node right){
        this.data = data;
        this.left = left;
        this.right = right;
    }
    public Node(int data){
        this(data,null,null);
    }
    public Node(){
        this(0,null,null);
    }


    //求该结点的高度
    public int getHeight(){
        return Math.max(this.left==null?0:this.left.getHeight(),this.right==null?0:this.right.getHeight())+1;
    }

    //左节点的高度
    public int getLeftHeight(){
        if(this.left==null)
            return 0;
        else
            return this.left.getHeight();
    }
    //右节点的高度
    public int getRightHeight(){
        if(this.right==null)
            return 0;
        else
            return this.right.getHeight();
    }

    //左旋
    public void leftRotate(){
        Node node=new Node(this.data);
        node.left=this.left;
        node.right=this.right.left;
        this.data=this.right.data;
        this.right=this.right.right;
        this.left=node;
    }

    //右旋
    public void rightRotate(){
        Node node=new Node(this.data);
        node.right=this.right;
        node.left=this.left.right;
        this.data=this.left.data;
        this.left=this.left.left;
        this.right=node;
    }

    /**
     * 向二叉排序树添加结点
     * @param node
     */
    public void add(Node node){
        if(node==null){
            return;
        }
        if(node.data<this.data){
            if(this.left==null){
                this.setLeft(node);
            }
            else{
                this.left.add(node);
            }
        }
        else{
            if(this.right==null){
                this.setRight(node);
            }
            else{
                this.right.add(node);
            }
        }

        //右子树的高度-左子树的高度)>1,左旋转
        if(this.getRightHeight()-this.getLeftHeight()>1){
            //如果它的右子树的左子树高度大于它的左子树高度,双旋转
            if(this.right!=null&&this.right.getLeftHeight()>this.getLeftHeight()){
                //先对这个结点的右节点进行右旋转
                this.right.rightRotate();
            }
            //左旋转
            this.leftRotate();
        }
        //左子树的高度-右子树的高度)>1,右旋转
        else if(this.getLeftHeight()-this.getRightHeight()>1){
            //如果它的左子树的右子树高度大于它的右子树高度,双旋转
            if(this.left!=null&&this.left.getRightHeight()>this.getRightHeight()){
                //先对这个结点的左节点进行左旋转
                this.left.leftRotate();
            }
            //右旋转
            this.rightRotate();
        }
        else{
            return;
        }
    }


}

测试类:

import java.util.*;
public class Main {
    public static void main(String[] args) {
        /**
         * 平衡二叉搜索树测试
         */
        Node root = new Node(25);
        root.add(new Node(18));
        root.add(new Node(16));
        root.add(new Node(19));
        root.add(new Node(29));
        root.add(new Node(27));
        root.add(new Node(30));
        root.add(new Node(31));
        root.add(new Node(32));

        //输出树
        System.out.println(Arrays.toString(levelOrder(root)));
        root.add(new Node(35));
        root.add(new Node(36));
        System.out.println(Arrays.toString(levelOrder(root)));
    }

    /**
     * 层序遍历
     * @param root
     * @return
     */
    public static int[] levelOrder(Node root) {
        if(root == null){
            return new int[0];
        }

        Queue<Node> queue = new LinkedList<Node>();
        queue.add(root);
        ArrayList<Integer> arr = new ArrayList<>();
        while( !queue.isEmpty() ){
            Node temp = queue.poll();
            arr.add(temp.getData());
            if(temp.getLeft() != null){
                queue.add(temp.getLeft());
            }
            if(temp.getRight() != null){
                queue.add(temp.getRight());
            }
        }
        int[] res = new int[arr.size()];
        for(int i = 0;i < arr.size();i++){
            res[i] = arr.get(i);
        }

        return res;

    }

}

执行结果:

*重点:

1,平衡二叉搜索树是一种高度平衡的二叉树;

2,由于对平衡的高要求,插入删除时会对树进行频繁的旋转操作。

 

标签:Node,right,Java,二叉,AVL,new,root,public,left
From: https://www.cnblogs.com/hangwei/p/16915790.html

相关文章

  • java中级考试
    选择2020分判断1010分解答55分程序页面+综合45分 第2章Css选择器CSS规则由三部分构成:选择符,属性和属性值   选择符{属性:属性值;属性:属性值;...}CSS......
  • Java工具库Guava的数学运算常用方法示例代码
    场景Java核心工具库Guava介绍以及Optional和Preconditions使用进行非空和数据校验:https://blog.csdn.net/BADAO_LIUMANG_QIZHI/article/details/127683387为什么使用Gu......
  • 如何通过Java 合并和取消合并 Excel 单元格
    在整理Excel中的数据时,我们不可避免地需要合并和取消合并单元格。同时,如果需要创建跨列或行的标题,我们可以合并Excel单元格以在电子表格中轻松完成此操作。合并单元格......
  • java执行流程
    编译:是指使用Java编译器对源文件进行错误排査的过程,编译后将生成后缀名为.class的字节码文件,不像C语言那样生成可执行文件。Java解释器负责将字节码文件翻译成具体......
  • JAVAEE
    javaee背景克服传统Client/Server架构的弊端迎合Browser/Serexer架构的潮流应用领域为中大型企业软件中大型网站B/S部署企业应用或者网站(联网)Web服务器......
  • JavaScript
    JavaScript内部标签<script>......</script>外部写法j.js<scriptsrc="js/j.js"></script>基础语法<script>//1.定义变量变量类型变量名=变量值......
  • java:绘制图形
    java绘图类:Graphics类 绘图是高级程序中必备的技术,在很多方面都能用到,如:绘制闪屏图片,背景图片和组件外观等。1.Graphics类 Graphics类是所有图形上下文的抽象基类,Gr......
  • java pdf 合并
    packagecom.hefeng.demo.controller;importjava.io.File;importjava.io.IOException;importjava.util.*;importorg.apache.pdfbox.io.MemoryUsageSetting;impo......
  • 【转载】Java List对象集合按对象属性分组、分组汇总、过滤等操作示例
    importjava.util.ArrayList;importjava.util.List;importjava.util.Map;importjava.util.stream.Collectors; publicclassTest{   publicstaticvoidmain(St......
  • 学习Java应该如何更快掌握
    学习Java    首先要想学Java,毋庸置疑的是,在你学习Java一定要有耐心。与此可以沉的下心更容易在其中钻研进步,Java学习只要找对方法难度都不是很高的,建议在学习前先了......