AVL树的构建

Posted lianzhen

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了AVL树的构建相关的知识,希望对你有一定的参考价值。

package com.xd.leetcode.shu;

/**
 * created by lianzhen on 2020-03-10 10:27. describe:平衡二叉树的构建
 * 
 * LL:插入的结点在左子树的左边导致失衡:右旋(顺时针旋转) 
 * RR: 插入的结点在右子树的右边导致失衡:左旋(逆时针方向)
 * LR:插入的结点在左子树的右边导致失衡:先按照根节点的左子树左旋再按照根节点进行右旋
 * RL:插入的结点在右子树的左边导致失衡:先按照根结点的右子树右旋再按照根节点进行左旋
 */
public class AVLTree {

  class Node {
    int data; //数据
    int height;//高度
    Node leftNode;//左节点
    Node rightNode;//右节点

    public Node(int data, Node leftNode, Node rightNode) {
      this.data = data;
      this.leftNode = leftNode;
      this.rightNode = rightNode;
    }
  }

  //计算每个根节点的高度
  private int getHeight(Node tagNode) {
    if (tagNode == null) {
      return -1;
    }
    return tagNode.height;
  }


  //右旋转
  private Node rightRotation(Node tagNode) {
    //找到这个目标节点的左子树的根节点
    Node leftChild = tagNode.leftNode;
    if (leftChild.rightNode != null) {
      //把自己的右子树作为目标节点的左子树
      tagNode.leftNode = leftChild.rightNode;
    }
    //让目标节点成为自己的右子树
    leftChild.rightNode = tagNode;
    //重新设置高度
    tagNode.height = Math.max(getHeight(tagNode.leftNode), getHeight(tagNode.rightNode)) + 1;
    leftChild.height = Math.max(getHeight(leftChild.leftNode), getHeight(leftChild.rightNode) + 1);
    return leftChild;
  }

  //左旋转
  private Node leftRotation(Node tagNode) {
    //找到这个目标节点的右子树的根节点
    Node rightNode = tagNode.rightNode;
    if (rightNode.leftNode != null) {
      //把自己的左子树作为目标节点的右子树
      tagNode.leftNode = rightNode.leftNode;
    }
    //让目标节点成为自己的左子树
    rightNode.leftNode = tagNode;

    tagNode.height = Math.max(getHeight(tagNode.leftNode), getHeight(tagNode.rightNode)) + 1;
    rightNode.height = Math.max(getHeight(rightNode.leftNode), getHeight(rightNode.rightNode)) + 1;
    return rightNode;
  }

  //左右旋转LR
  private Node rightLeftRatation(Node tagNode) {
    //先按照目标节点的左子树的根节点进行右旋转
    Node leftChild = tagNode.leftNode;
    leftRotation(leftChild);
    //再按照目标节点进行左旋转
   return rightRotation(tagNode);
  }

  //右左旋转
  private Node leftRightRatation(Node tagNode) {
    //先按照目标节点的右子树的根节点进行左旋转
    Node rightChild = tagNode.rightNode;
    rightRotation(rightChild);
    //再按照目标节点进行右旋转
   return leftRotation(tagNode);
  }

  /**
   * 朝根的节点下方插入新的节点 不需要要考虑那么多 只考虑当前的节点再上一个节点的左边或者右边就行
   *
   * @param root 根节点
   * @param data 需要插入节点的数据
   */
  private Node insertNode(Node root, int data) {
    if (root == null) {
      root = new Node(data, null, null);
    }else {
      if (data <= root.data) {
        //把新的节点放在这个节点的左边
        root.leftNode = insertNode(root.leftNode, data);
        //判断左右子树的高度
        if(getHeight(root.leftNode)-getHeight(root.rightNode)>1){
          if(data<=root.leftNode.data){
            //插入的位置是左子树的左边 ---进行右旋转  
            root= rightRotation(root);
          }else {
            //插入的位置是左子树的右边LR 先左后右旋
            root=leftRightRatation(root);
          }
        }

      } else {
        //把新的节点放在这个节点的右边
        root.rightNode = insertNode(root.rightNode, data);
        //判断左右子树的高度
        if(getHeight(root.rightNode)-getHeight(root.leftNode)>1){
          if(data<=root.leftNode.data){
            //进行右左旋转
            root= rightLeftRatation(root);
          }else {
            //进行右旋转
            root=rightRotation(root);
          }
        }
      }
    }

    //对节点的高度进行更新(有可能不变)
    root.height=Math.max(getHeight(root.leftNode),getHeight(root.rightNode))+1;
    return root;
  }

}

 

以上是关于AVL树的构建的主要内容,如果未能解决你的问题,请参考以下文章

C++AVL树的实现--详细解析旋转细节

C++AVL树的实现--详细解析旋转细节

数据结构之树篇3——平衡二叉树(AVL树)

小代码 向原文学习 对AVL树的4种情况 用字母标记整理

❤️数据结构入门❤️(2 - 2)- AVL 树

徒手写的AVL竟然比STL中的红黑树效率更高?✨