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树的构建的更多相关文章

  1. 二叉树与AVL树

    二叉树 什么是二叉树? 父节点至多只有两个子树的树形结构成为二叉树.如下图所示,图1不是二叉树,图2是一棵二叉树. 图1 普通的树                                    ...

  2. 数据结构图文解析之:AVL树详解及C++模板实现

    0. 数据结构图文解析系列 数据结构系列文章 数据结构图文解析之:数组.单链表.双链表介绍及C++模板实现 数据结构图文解析之:栈的简介及C++模板实现 数据结构图文解析之:队列详解与C++模板实现 ...

  3. 平衡搜索树(一) AVL树

    AVL树 AVL树又称为高度平衡的二叉搜索树,是1962年有俄罗斯的数学家G.M.Adel'son-Vel'skii和E.M.Landis提出来的.它能保持二叉树的高度 平衡,尽量降低二叉树的高度,减 ...

  4. 数据结构与算法(九):AVL树详细讲解

    数据结构与算法(一):基础简介 数据结构与算法(二):基于数组的实现ArrayList源码彻底分析 数据结构与算法(三):基于链表的实现LinkedList源码彻底分析 数据结构与算法(四):基于哈希 ...

  5. AVL树和平衡二叉树 平衡因子 右旋转LL 左旋转RR LR RL

    前言 今天要介绍几种高级数据结构AVL树,介绍之前AVL,会先说明平衡二叉树,并将树的学习路线进行总结,并介绍维持平衡的方法:右旋转.左旋转. 一.树学习路线 1.路线总结 总结了一下树的学习路线,如 ...

  6. 二叉查找树(BST)、平衡二叉树(AVL树)(只有插入说明)

    二叉查找树(BST).平衡二叉树(AVL树)(只有插入说明) 二叉查找树(BST) 特殊的二叉树,又称为排序二叉树.二叉搜索树.二叉排序树. 二叉查找树实际上是数据域有序的二叉树,即对树上的每个结点, ...

  7. 二叉查找树(BST)、平衡二叉树(AVL树)

    二叉查找树(BST) 特殊的二叉树,又称为排序二叉树.二叉搜索树.二叉排序树. 二叉查找树实际上是数据域有序的二叉树,即对树上的每个结点,都满足其左子树上所有结点的数据域均小于或等于根结点的数据域,右 ...

  8. 【Java】 大话数据结构(12) 查找算法(3) (平衡二叉树(AVL树))

    本文根据<大话数据结构>一书及网络资料,实现了Java版的平衡二叉树(AVL树). 平衡二叉树介绍 在上篇博客中所实现的二叉排序树(二叉搜索树),其查找性能取决于二叉排序树的形状,当二叉排 ...

  9. 树-二叉搜索树-AVL树

    树-二叉搜索树-AVL树 树 树的基本概念 节点的度:节点的儿子数 树的度:Max{节点的度} 节点的高度:节点到各叶节点的最大路径长度 树的高度:根节点的高度 节点的深度(层数):根节点到该节点的路 ...

  10. AVL树 算法思想与代码实现

    AVL树是高度平衡的二叉搜索树,按照二叉搜索树(Binary Search Tree)的性质,AVL首先要满足: 若它的左子树不为空,则左子树上所有结点的值均小于它的根结点的值: 若它的右子树不为空, ...

随机推荐

  1. Verilog中的时间尺度与延迟

    在Verilog的建模中,时间尺度和延迟是非常重要的概念,设置好时间尺度和延迟,可以充分模拟逻辑电路发生的各种情况和事件发生的时间点,来评估数字IC设计的各种要求,达到充分评估和仿真的作用.注意延迟语 ...

  2. cesium 學習計劃

    上篇已经将cesium环境搭建完成,并且服务启动完成,进去后主要浏览 documents 文档和Sandcastle 示例. 学习计划:沙盒示例学习一遍,每个示例中的查看对应代码接口. 学习cesiu ...

  3. Modelsim中的Verilog语言使用

    一.建立工程 1.在建立工程(project)前,先建立一个工作库(library),一般将这个 library 命名为 work.尤其是第一次运行 modelsim 时,是没有这个"wor ...

  4. Crypto入门 (四)不仅仅是Morse(Morse+Bacon)

    前言: 前面我们刚刚学过了解Morsecode,现在我们将继续学习 不仅仅是Morse: 题目:--/.-/-.--/..--.-/-..././..--.-/..../.-/...-/./..--. ...

  5. 记录一次MySQL主从同步

    主库配置 server-id=1log-bin=mysql-binbinlog_format=ROWbinlog_row_image=minimalbinlog-do-db=yjtb-cloud 解释 ...

  6. I - Cloud Retainer's Game

    I - Cloud Retainer's Game 传送门: I. Cloud Retainer's Game (codeforces.com) 题意: 在坐标轴上有2个边界:y=0和y=H.有n个质 ...

  7. loadrunner写webservice接口

    先用soupUI调试  fiddler抓包 然后再写: web_custom_request("createSoapOrder",         "URL=http:/ ...

  8. 单例和mono单例

    单例 public class Singleton<T> where T : new() { private static T instance; public static T Inst ...

  9. 给c++写python的split()与input()【python一样写c++、一】

    python的split确实是很香的功能. 写c++的时候,就会想着,要是能直接input().split()那不挺好. 实际上真的可以:自己动手,丰衣足食. 先放成品展示. int main(){ ...

  10. c++的thread小测试

    windows环境还用不了thread,得下一些mingw,弄了半天没弄好,直接用了商店中心就有的Ubuntu了,但是sudo install g++出现了下载不了的问题,解决方案:https://b ...