Java数据结构(十四)—— 平衡二叉树(AVL树)
平衡二叉树(AVL树)
二叉排序树问题分析
左子树全部为空,从形式上看更像一个单链表
插入速度没有影响
查询速度明显降低
解决方案:平衡二叉树
基本介绍
平衡二叉树也叫二叉搜索树,保证查询效率较高
它是一棵空树或它的左右两个子树的高度差的绝对值不超过1,并且左右两棵子树都是一棵平衡二叉树
常用的实现方法有红黑树、AVL、替罪羊树、Treap、伸展树等
平衡二叉树左旋转
使用条件
右子树高度与左子树高度插值大于1的时候,使用左旋转
要求
给定数列{4,3,6,5,7,8},创建对应的平衡二叉树
创建二叉排序树
此时若转换为平衡二叉树,降低右子树的高度
思路分析
创建一个新节点newNode ,值等于当前根节点的值
把新节点的左子树设置为当前节点的左子树,newnode.left = left
把新结点的右子树设置为当前节点右子树的左子树newNode.right = right.left
把当前节点的值换位右子节点的值 value = right.value
把当前节点的右子树设置成右子树的右子树right = right.right
把当前节点的左子树设置为新节点 left = newLeft
转换后
平衡二叉树右旋转
要求
使用数列{10,12,8,9,7,6},创建平衡二叉树
创建二叉排序树
基本思路
创建新的节点newNode,使得newNode.value = this.value
将newNode的右子树设置为this的右子树,newNode.right = this.right
将newNode的左子树设置为this左子树的右子树,newNode.left = this.left.right
把this节点的值换为左子节点的值,this.value = this.left.value
将this节点的左子树设置为左子树的左子树,this.left = this.left.left
将this节点的右子树 设置为newNode,this.right = newNode
转换后
平衡二叉树双旋转
要求
使用数列{10,11,7,6,8,9},创建平衡二叉树
创建二叉排序树
思路分析
当孩子节点满足左旋转或右旋转条件时,先平衡孩子 节点,后平衡父节点
创建平衡二叉树代码实现
package com.why.tree;
/**
* @Description TODO 平衡二叉树
* @Author why
* @Date 2020/12/6 15:56
* Version 1.0
**/
public class AVLTreeDemo {
public static void main(String[] args) {
int[] arr = {10,11,7,6,8,9};
//创建AVLTree对象
AVLTree avlTree = new AVLTree();
//添加节点
for (int i = 0; i < arr.length; i++) {
avlTree.add(new AVLNode(arr[i]));
}
//遍历
System.out.println("中序遍历:");
avlTree.midOrder();
//根节点树的高度
System.out.println("根节点树的高度: " + avlTree.height());
System.out.println("左子树高度:" + avlTree.leftHeight());
System.out.println("右子树高度:" + avlTree.rightHeight());
System.out.println("根节点:" + avlTree.getRoot());
}
}
/**
* AVL,平衡二叉树
*/
class AVLTree{
private AVLNode root;
public AVLNode getRoot() {
return root;
}
public void setRoot(AVLNode root) {
this.root = root;
}
/**
* 添加节点
* @param node
*/
public void add(AVLNode node){
if (root == null){//直接放上
root = node;
}else {
root.add(node);
}
}
/**
* 中序遍历
*/
public void midOrder(){
if (root != null){
root.midOrder();
}else {
System.out.println("二叉排序树为空");
}
}
/**
* 查找需删除的节点
* @param value
* @return
*/
public AVLNode search(int value){
if (root == null){
return null;
}else {
return root.search(value);
}
}
/**
* 查找父节点
* @param value
* @return
*/
public AVLNode searchParent(int value){
if (root == null){
return null;
}else {
return root.searchParent(value);
}
}
public void deleteNode(int value){
if (root == null){
return;
}else {
//找到需删除的节点
AVLNode targetNode = search(value);
if (targetNode == null){//未找到
return;
}
//如果二叉排序树只有一个节点
if (root.left == null && root.right == null){
return;
}
//查找需删除的节点的父节点
AVLNode parent = searchParent(value);
if (targetNode.left == null && targetNode.right == null){//删除的节点是叶子节点
//判断targetNode是父节点的左子节点还是右子节点
if (parent.left != null && parent.left.value == value){//是左子节点
parent.left = null;
}else if (parent.right != null && parent.right.value == value){//是右子节点
parent.right = null;
}
}else if ((targetNode.left != null && targetNode.right == null) ||
(targetNode.right != null && targetNode.left == null)) {//只有一棵子树
//确定targetNode的节点是左节点还是右节点
if (targetNode.left != null) {//左子节点
if (parent != null){//非根节点
//确定targetNode是parent的左子节点还是右子节点
if (parent.left.value == value) {//左子节点
parent.left = targetNode.left;
} else {//右子节点
parent.right = targetNode.left;
}
}else {
root = targetNode.left;
}
} else {//右子节点
if (parent != null){
//确定targetNode是parent的左子节点还是右子节点
if (parent.left.value == value) {//左子节点
parent.left = targetNode.right;
} else {//右子节点
parent.right = targetNode.right;
}
}else {
root = targetNode.right;
}
}
}else {//删除的节点有两颗子树
//找到最小值并删除
int minValue = deleteRightMin(targetNode.right);
//将最小值赋值给targetNode.value
targetNode.value = minValue;
}
}
}
/**
* 寻找最小值
* @param node
* @return
*/
public int deleteRightMin(AVLNode node){
AVLNode target = node;
while (target.left != null){
target = target.left;
}
//这时target指向最小节点
//删除最小节点
deleteNode(target.value);
//返回最小节点的value
return target.value;
}
/**
* 返回根节点树的高度
* @return
*/
public int height(){
return root.height();
}
/**
* 左子树的高度
* @return
*/
public int leftHeight(){
return root.leftHeight();
}
/**
* 右子树的高度
* @return
*/
public int rightHeight(){
return root.rightHeight();
}
}
/**
* 节点类
*/
class AVLNode{
int value;
AVLNode left;
AVLNode right;
public AVLNode(int value) {
this.value = value;
}
/**
* 添加节点,递归形式,需满足二叉排序树的要求
* @param node
*/
public void add(AVLNode node){
if (node == null){
return;
}
//判断传入的节点的值和当前子树的根节点的值的关系
if (node.value < this.value){
if (this.left == null){//当前节点左子节点为空
this.left = node;
}else {//不为空,递归向左子树添加
this.left.add(node);
}
}else {
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();
}
//左旋转
this.leftRotate();
return;
}
//当添加完节点后leftHeight - rightHeight > 1
if (leftHeight() - rightHeight() > 1){
if (left != null && left.rightHeight() > left.leftHeight()){
//对左子树左旋转
left.leftRotate();
}
//右旋转
this.rightRotate();
return;
}
}
/**
* 中序遍历
*/
public void midOrder(){
if (left != null){
this.left.midOrder();
}
System.out.println(this);
if (this.right != null){
this.right.midOrder();
}
}
@Override
public String toString() {
return "Node{" +
"value=" + value +
'}';
}
/**
* 寻找需要删除的节点
* @param value
* @return
*/
public AVLNode search(int value){
if (value == this.value){//找到
return this;
}else if (value < this.value){//向左子树查找
if (this.left == null){
return null;
}
return this.left.search(value);
}else {//向右子树查找
if (this.right == null){
return null;
}
return this.right.search(value);
}
}
/**
* 查找需要删除节点的父节点
* @param value
* @return
*/
public AVLNode searchParent(int value){
if ((this.left != null && this.left.value == value) || (this.right != null && this.right.value == value)){
//找到父节点返回当前节点
return this;
}else {
//如果查找的值小于当前节点的值
if (value < this.value && this.left != null){//左子树查找
return this.left.searchParent(value);
}else if (value >= this.value && this.right != null){//右子树查找
return this.right.searchParent(value);
}else {
return null;//没有找到父节点
}
}
}
/**
* 返回以当前节点为根节点的树的高度
* @return
*/
public int height(){
return Math.max(this.left == null ? 0 : this.left.height(),this.right == null ? 0 : this.right.height()) + 1;
}
/**
* 返回左子树的高度
* @return
*/
public int leftHeight(){
if (left == null){
return 0;
}else {
return left.height();
}
}
/**
* 返回右子树的高度
* @return
*/
public int rightHeight(){
if (right == null){
return 0;
}else {
return right.height();
}
}
/**
* 左旋转方法
*/
private void leftRotate(){
//创建新的节点,以当前根节点的值创建
AVLNode newNode = new AVLNode(this.value);
//把新的节点的左子树设置为当前节点的左子树
newNode.left = this.left;
//把新节点的右子树设置为当前节点右子树的左子树
newNode.right = this.right.left;
//将当前节点的值修改为右子树的值
this.value = this.right.value;
//将当前节点的右子树设置为右子树的右子树
this.right = this.right.right;
//将当前节点的左子节点设置为新的节点
this.left = newNode;
}
/**
* 右旋转
*/
private void rightRotate(){
//以当前节点的值创建新的节点
AVLNode newNode = new AVLNode(this.value);
//将新节点的右子树设置为当前节点的右子树
newNode.right = this.right;
//将当前节点的左子树设置为当前节点左子节点的右子树
newNode.left = this.left.right;
//将当前节点的值用左子节点的值替换
this.value = this.left.value;
//将当前节点的左子节点设置为当节点左子节点的左子树
this.left = this.left.left;
//将当前节点的右子节点设置为新节点
this.right = newNode;
}
}
所有源码都可在gitee仓库中下载:https://gitee.com/vvwhyyy/java_algorithm
Java数据结构(十四)—— 平衡二叉树(AVL树)的更多相关文章
- Java数据结构和算法(七)--AVL树
在上篇博客中,学习了二分搜索树:Java数据结构和算法(六)--二叉树,但是二分搜索树本身存在一个问题: 如果现在插入的数据为1,2,3,4,5,6,这样有序的数据,或者是逆序 这种情况下的二分搜索树 ...
- 自己动手实现java数据结构(七) AVL树
1.AVL树介绍 前面我们已经介绍了二叉搜索树.普通的二叉搜索树在插入.删除数据时可能使得全树的数据分布不平衡,退化,导致二叉搜索树最关键的查询效率急剧降低.这也引出了平衡二叉搜索树的概念,平衡二叉搜 ...
- Java 树结构实际应用 四(平衡二叉树/AVL树)
平衡二叉树(AVL 树) 1 看一个案例(说明二叉排序树可能的问题) 给你一个数列{1,2,3,4,5,6},要求创建一颗二叉排序树(BST), 并分析问题所在. 左边 BST 存在的问题分析: ...
- 二叉查找树(BST)、平衡二叉树(AVL树)(只有插入说明)
二叉查找树(BST).平衡二叉树(AVL树)(只有插入说明) 二叉查找树(BST) 特殊的二叉树,又称为排序二叉树.二叉搜索树.二叉排序树. 二叉查找树实际上是数据域有序的二叉树,即对树上的每个结点, ...
- “全栈2019”Java第九十四章:局部内部类详解
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- “全栈2019”Java第十四章:二进制、八进制、十六进制
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- “全栈2019”Java第二十四章:流程控制语句中决策语句switch下篇
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 IntelliJ IDEA v2018.3 文章原文链接 "全栈2019"Java第 ...
- 【数据结构】平衡二叉树—AVL树
(百度百科)在计算机科学中,AVL树是最先发明的自平衡二叉查找树.在AVL树中任何节点的两个子树的高度最大差别为一,所以它也被称为高度平衡树.查找.插入和删除在平均和最坏情况下都是O(log n).增 ...
- 数据结构快速回顾——平衡二叉树 AVL (转)
平衡二叉树(Balanced Binary Tree)是二叉查找树的一个进化体,也是第一个引入平衡概念的二叉树.1962年,G.M. Adelson-Velsky 和 E.M. Landis发明了这棵 ...
- 平衡二叉树,AVL树之图解篇
学习过了二叉查找树,想必大家有遇到一个问题.例如,将一个数组{1,2,3,4}依次插入树的时候,形成了图1的情况.有建立树与没建立树对于数据的增删查改已经没有了任何帮助,反而增添了维护的成本.而只有建 ...
随机推荐
- Docker(4)- Docker 命令大全
如果你还想从头学起 Docker,可以看看这个系列的文章哦! https://www.cnblogs.com/poloyy/category/1870863.html 容器生命周期管理 run sta ...
- Spark编程练习题
import org.apache.spark.sql.SparkSessionval spark = SparkSession.builder().appName("Spark SQL b ...
- 性能问题,AWR High Event enq: US - contention
1.1问题现象 应用反馈业务执行SQL响应超时,需要数据库排除DB是否存在问题,创建AWR观察到top event 新增enq: US - contention ??? 1.2问题分析 1) DB ...
- 华为+京东数科(原京东金融)面经--Java后台开发
华为: 1.笔试中遇到的问题,如何解决的?(Scanner 如何结束循环读取数据,笔者在面试中因没有理解到Scanner类的hasNext()与hasNextLine()是阻塞方法,导致没有正确退出循 ...
- 采用“传统”方式获取当前HttpContext
我们知道"依赖注入"已经成为了.NET Core的基本编程模式,表示当前请求上下文的HttpContext可以通过注入的IHttpContextAccessor服务来提取.有时候我 ...
- 【JVM第四篇--运行时数据区】堆
写在前面的话:本文是在观看尚硅谷JVM教程后,整理的学习笔记.其观看地址如下:尚硅谷2020最新版宋红康JVM教程 一.堆的概述 JVM的运行时数据区如下: 一个Java程序运行起来对应着一个进程(操 ...
- tensorflow的tfrecord操作代码与数据协议规范
tensorflow的数据集可以说是非常重要的部分,我认为人工智能就是数据加算法,数据没处理好哪来的算法? 对此tensorflow有一个专门管理数据集的方式tfrecord·在训练数据时提取图片与标 ...
- 企业网络拓扑VRRP主备功能实例(一)
组网图形 VRRP主备备份简介 通常,同一网段内的所有主机上都存在一条相同的.以网关为下一跳的缺省路由.主机发往其他网段的报文将通过缺省路由发往网关,再由网关进行转发,从而实现主机与外部网络的通信. ...
- 头秃了,二十三张图带你从源码了解Spring Boot 的启动流程~
持续原创输出,点击上方蓝字关注我 目录 前言 源码版本 从哪入手? 源码如何切分? 如何创建SpringApplication? 设置应用类型 设置初始化器(Initializer) 设置监听器(Li ...
- du查看的目录大小与df查看的大小不同的时候用lsof查找
首先MAN一下两个命令,看一下解释的区别: du - estimate file space usage df - report file system disk space usage du估计文件 ...