平衡二叉树(AVLTree)

AVLTree

  • 1、树的分类
  • 2、平衡二叉树
    • 2.1、构建一个平衡二叉树
    • 2.2、删除节点
    • 2.3、搜索方式
      • 2.3.1、广度优先搜索(BFS)
      • 2.3.2、深度优先搜索(DFS)

1、树的分类

树形结构是编程当中特别常见的一种数据结构。比如电脑中的文件管理系统就大量用到树结构。
![在这里插入图片描述](https://img-
##blog.csdnimg.cn/direct/2f412f2f7b144da3b4193557dc388274.png#pic_center =600x400)
该图取之于这篇文章 硬核总结!真二叉树、满二叉树、完全二叉树的性质与概念,这篇文章对一些基础的树总结的很好。所以我也不赘诉了。

2、平衡二叉树

平衡二叉树的基础是二叉搜索树,对于一个二叉搜索树而言,新插入的数字若小于根节点则放在左边,反之放在右边。但如果插入的数字一直小于或大于根节点,那这个树,就和链表差不多了。如图,我按顺序插入(1, 0, 3, 2, 4, 5, 6)。
在这里插入图片描述
本来二叉树的实现就是为了减低搜索的时间复杂度的,而如果树构建得和单链表差不多,那便失去了意义。所以二叉平衡搜索树,又称平衡二叉树应运而生。
平衡二叉树要求左子树的高度和右子树的高度之不能超过1。

2.1、构建一个平衡二叉树

正如前面所言,一个平衡二叉树,首先得是一个二叉搜索树,然后再要求左右子树高度差不超一。于是人们总结出四种会导致不平衡的情况。
1、LL型:即左子树的左子树增高了一层,从而导致不平衡。
在这里插入图片描述
上图是一个简单的模型,实际上为了解决不平衡的问题,要解决的事情远不止怎么简单。下面是一个更为通用的模型。我不仅需要把a变成b的右子树,还需要把原来b的右子树变成a的左子树,然后还要让原来指向a的指针指向b。这是一个颇为麻烦的事情,最难的部分是要改变根节点上级指针。
在这里插入图片描述
2、LR型:左子树的右子树添加了一层导致不平衡。直接上图。
在这里插入图片描述
另外两种类型就是RR和RL了,逻辑和上述两种都是一样的。再写代码的时候甚至可以把所以left和right调换然后就可以实现LL到RR的转换或LR到RL的转换。而我懒得画图就不再画了。
直接上代码。

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;
#define n 9struct Node
{int data;Node* right;Node* left;Node() {};Node(int x): data(x) {};            //constructor
};class AVLTree
{
public:Node* root;AVLTree(){root = nullptr;};int get_height(Node* node);void append_data(Node** node, int num);       //append a number into the AVL treevoid rrRotation(Node** root);void rlRotation(Node** root);void llRotation(Node** root);void lrRotation(Node** root);
};int AVLTree::get_height(Node* node)
{if(node == nullptr) return 0;return max(get_height(node->left), get_height(node->right)) + 1;
}void AVLTree::append_data(Node** node, int data)
{if(*node == nullptr){*node = new Node(data);(*node)->right = nullptr;(*node)->left = nullptr;}else if(data < (*node)->data){append_data(&(*node)->left, data);if(abs(get_height((*node)->right) - get_height((*node)->left)) > 1){if(data < (*node)->left->data) llRotation(node);else lrRotation(node);}}else{append_data(&(*node)->right, data);if(abs(get_height((*node)->right) - get_height((*node)->left)) > 1){if(data > (*node)->right->data) rrRotation(node);else rlRotation(node);}}
}void AVLTree::llRotation(Node** root)
{Node* temp = (*root)->left;(*root)->left = temp->right;temp->right = *root;*root = temp;
}void AVLTree::lrRotation(Node** root)
{Node* temp = (*root)->left->right;(*root)->left->right = temp->left;temp->right = *root;Node* temp2 = temp->left;temp->left = (*root)->left;(*root)->left = temp2;*root = temp;
}void AVLTree::rrRotation(Node** root)
{Node* temp = (*root)->right;(*root)->right = temp->left;temp->left = *root;*root = temp;
}void AVLTree::rlRotation(Node** root)
{Node* temp = (*root)->right->left;(*root)->right->left = temp->right;temp->left = *root;Node* temp2 = temp->right;temp->right = (*root)->right;(*root)->right = temp2;*root = temp;
}int randomArray[n] = {16, 3, 7, 11, 9, 26, 18, 14, 15};int main()
{class AVLTree mytree;for(int i = 0; i < n; i++)mytree.append_data(&mytree.root, randomArray[i]);printf("the height of tree is: %d\n", mytree.get_height(mytree.root));system("pause");return 0;
}

在这个代码实现中,最令我头疼的是二级指针这里。我一开始不太明白用二级指针的意义,然后尝试了许多次才不得不使用它。
因为在我插入一个节点之后,我是通过反向遍历的方式去查找哪个节点失衡了。但是此时有一个问题,当我得知这个节点失衡之后,我并不清楚它的上一个节点再哪里,因此我无法改变上一个节点所存储的指针。有点绕,我画个图。
如下面两个图所示,如果b这个节点失衡了,但又因为我用的是反向遍历,所以我并不知道a这个节点的地址。故而我无法改变a->right的指向。而为了解决这个问题,我不得不引入二级指针,让我得到b的地址的同时,也得知a.right的地址。
在这里插入图片描述
在这里插入图片描述
具体逻辑是任何对b的操作,都是先取得&a.right再解引用。这样相当于知道b便知道a.right的地址。
在这里插入图片描述
以上,如果不动手实践的话,大概是看不明白的。。。

2.2、删除节点

下面这个图表红的三个框,分别对应着删除一个节点的三种情况。
情况一:节点位于末端,直接删除即可;
情况二:节点有一个孩子,那么删除之后需要重新建立起父子连接;
情况三:节点有两个孩子,此时有两种策略。第一种策略是把右孩子,即右子树的最小值复制到自己身上。然后把被复制的那个节点删除。第二种策略是把左子树的最大值复制到自己身上,然后把被复制的那个节点删除。
在这里插入图片描述
下面这个图是遇到情况三之后采取策略一的示意图。看到这里,可能许多人还存在一个疑惑,难道不怕2下面还有孩子吗?答案是否定的,因为右子树的最小值,即为最值,自然就是没有孩子的节点了。所以此时删除2,只需要切换到情况一即可。
在这里插入图片描述

2.3、搜索方式

2.3.1、广度优先搜索(BFS)

如图,广度优先搜索是从根节点到末节点,一层一层这些遍历下去的。实现起来也并不复杂,只需要设置一个FIFO容器,比如queue。不断的从容器中读取节点即可。
在这里插入图片描述
queue内部的示意图大概是这样,但要注意,其实这些值不会同时出现在queue当中。
在这里插入图片描述
代码实现如下:

void bfs(Node** root)
{Node* temp = nullptr;queue<Node*> Q;if(*root != nullptr)   Q.push(*root);else return;while(!Q.empty()){temp = Q.front();if(temp->left != nullptr)    Q.push(temp->left);if(temp->right != nullptr)    Q.push(temp->right);cout << temp->data << endl;···Q.pop();}
}

2.3.2、深度优先搜索(DFS)

深度优先搜索分为前序遍历、中序遍历和后序遍历。主要还是依靠递归来实现。在代码上只是改动输出值的位置,相对容易。

void dfs_preorder(Node** node)			//前序
{if(*node == nullptr) return;cout << (*node)->data << endl;dfs_preorder(&((*node)->left));dfs_preorder(&((*node)->right));
}void dfs_inorder(Node** node)			//中序
{if(*node == nullptr) return;dfs_preorder(&((*node)->left));cout << (*node)->data << endl;dfs_preorder(&((*node)->right));
}void dfs_postorder(Node** node)		//后序
{if(*node == nullptr) return;dfs_preorder(&((*node)->left));dfs_preorder(&((*node)->right));cout << (*node)->data << endl;
}

分别看一下前序、中序、后序的打印顺序吧。注意,在以下这些图中,节点中的数字代表的是打印顺序。

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

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://xiahunao.cn/news/2979738.html

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

信息打点--公众号服务

微信公众号 获取微信公众号的途径https://weixin.sogou.com/ 微信公众号没有第三方服务 Github监控 人员&域名&邮箱 eg&#xff1a;xxx.cn password in:file https://gitee.com/ https://github.com/ https://www.huzhan.com/ 资源搜索 in:name test 仓库标题搜索含有…

C语言中字符串函数以及内存函数的使用和注意事项

目录 0. 前言 1、求字符串长度函数 1.1、strlen 模拟实现 2.长度不受限制的字符串函数 2.1 strcpy 模拟实现 2.2strcat 模拟实现 2.3strcmp 模拟实现 3.长度受限制的字符串函数 3.1strncpy 3.2strncat 3.3strncmp 4、字符串查找函数 4.1strstr 模拟实现 3.2strt…

修复版最新精仿熊猫办公PPT模板图片素材整站源码+WAP手机端+会员系统+火车头采集

修复版最新精仿熊猫办公PPT模板图片素材整站源码WAP手机端会员系统火车头&#xff0c;采用Empirecms7.5版UTF-8开发&#xff0c;是一款非常高端的ppt模板&#xff0c;图片素材下载站模板非常适合大型图库下载站&#xff0c;配有手机端模板&#xff0c;支持自定义设置会员组&…

面试官:在原生input上面使用v-model和组件上面使用有什么区别?

前言 还是上一篇面试官&#xff1a;来说说vue3是怎么处理内置的v-for、v-model等指令&#xff1f; 文章的那个粉丝&#xff0c;面试官接着问了他另外一个v-model的问题。 面试官&#xff1a;vue3的v-model都用过吧&#xff0c;来讲讲。 粉丝&#xff1a;v-model其实就是一个语…

python使用redis存储时序数据

import redisdef ts_demo():"""时序数据存储RedisTimeSeries测试"""# 连接到Redisr redis.Redis(hostlocalhost, password"xxxx", port63790, db0)r1 r.ts()# print(r1.get("ts_key"))# print(r.exists(ts_key))# # 清空键…

【技巧】Git 版本控制工具没有图标提示怎么办?

Git 版本控制工具在日常开发中使用率是非常高的&#xff0c;多数情况下会安装 TortoiseGit 之类的插件&#xff0c;让文件夹显示图标&#xff0c;方便观察文件的状态。但是有时装完插件之后发现&#xff0c;文件夹/文件并没有图标显示&#xff0c;可以按照以下思路进行排查&…

Ts支持哪些类型和类型运算(上)

目录 1、元组 2、接口&#xff08;interface&#xff09; 3、枚举&#xff08;Enum&#xff09; 4、字面量类型 5、keyof 6、in keyof 7、类型的装饰 静态类型系统 就是把 类型检查从运行时提前到了编译时&#xff0c;所以ts类型系统中的许多类型与js并无区别 例如&am…

2024-4-22 群讨论:微服务启动预热相关

以下来自本人拉的一个关于 Java 技术的讨论群。关注公众号&#xff1a;hashcon&#xff0c;私信进群拉你 Hotspot JVM 进程启动后&#xff0c;流量到来的时候 JIT 吃掉很多 CPU&#xff0c;如何观察到&#xff1f; 很多途径都能观察到&#xff1a; top -Hp&#xff1a;这个需…

MTK6775/MT6775/曦力P70联发科处理器性能参数资料

联发科MT6775(曦力P70)芯片搭载强大的Arm Cortex-A73/A53八核CPU&#xff0c;并采用台积电12纳米FinFET制程工艺&#xff0c;相较于其他14纳米级别产品&#xff0c;功耗节省达到了15%。此外&#xff0c;曦力P70还配备了高效能的Arm Mali-G72 GPU&#xff0c;相比上一代产品曦力…

Java——继承与组合

和继承类似, 组合也是一种表达类之间关系的方式, 也是能够达到代码重用的效果。组合并没有涉及到特殊的语法 (诸如 extends 这样的关键字), 仅仅是将一个类的实例作为另外一个类的字段。 继承表示对象之间是is-a的关系&#xff0c;比如&#xff1a;狗是动物&#xff0c;猫是动…

YOLOv9训练结果分析->mAP、Precision、Recall、FPS、Confienc、混淆矩阵分析

简介 这篇博客&#xff0c;主要给大家讲解我们在训练yolov9时生成的结果文件中各个图片及其中指标的含义&#xff0c;帮助大家更深入的理解&#xff0c;以及我们在评估模型时和发表论文时主要关注的参数有那些。本文通过举例训练过程中的某一时间的结果来帮助大家理解&#xf…

动态规划——切割钢条问题

一、动态规划 动态规划算法通常用于解决最优化问题&#xff08;寻求最优解&#xff09;。其思想与分治法类似&#xff0c;将待求解的问题分成若干个子问题&#xff0c;先求出子问题&#xff0c;再根据子问题的解求出原来问题中的解&#xff0c;与分支法不同的是&#xff0c;在动…

【勒索病毒恢复】.svh勒索病毒介绍及恢复方案

一、.[[backupwaifu.club]].svh勒索病毒介绍 svh勒索病毒是一种恶意软件&#xff0c;它通过加密受害者的文件并要求支付赎金来解锁&#xff0c;从而达到勒索的目的。这种病毒已经存在了数年&#xff0c;并且不断演变&#xff0c;形成了多种不同的家族和变种。如果您的数据承载着…

Bentley二次开发教程02-开发环境搭建

1 Bentley 平台介绍 图 1 Bentley 平台介绍 Bentley 软件大致可分为四大平台&#xff0c;分别为用于设计的 Microstation 平台&#xff0c;用于协同的 ProjectWise 平台&#xff0c;用于对资产进行全生命周期管理的 AssetWise 平台和数据互联互通的 数字孪生平台 iTwin。 1.1 …

CyclicBarrier(循环屏障)源码解读与使用

&#x1f3f7;️个人主页&#xff1a;牵着猫散步的鼠鼠 &#x1f3f7;️系列专栏&#xff1a;Java全栈-专栏 &#x1f3f7;️个人学习笔记&#xff0c;若有缺误&#xff0c;欢迎评论区指正 目录 1. 前言 2. 什么是CyclicBarrier&#xff1f; 3. CyclicBarrier与CountDownL…

潜力与限制:低代码开发平台优缺点全面分析

低代码开发平台作为一种创新技术工具&#xff0c;正以其快速开发、低门槛参与和灵活定制等特性&#xff0c;重塑企业数字化转型之路。然而&#xff0c;任何技术都有其两面性&#xff0c;低代码平台也不例外。本文将深入探讨低代码开发平台的优缺点&#xff0c;并为您推荐值得信…

git常见命令(成长版)

ps&#xff1a;所谓成长版就是后续可能还会添加命令&#xff1a; 1.删除本地分支&#xff1a; git branch -d 分支名 2.拉取代码后默认master分支&#xff0c;切换到线上其他分支&#xff1a; &#xff08;1&#xff09;查看线上所有分支&#xff1a; git branch -a &#…

排序算法:顺序查找

简介 顺序查找&#xff08;也称为线性查找&#xff09;是一种简单直观的搜索算法。按照顺序逐个比较列表或数组中的元素&#xff0c;直到找到目标元素或搜索完整个列表。 应用场景 数据集比较小&#xff0c;无需使用复杂的算法。数据集没有排序&#xff0c;不能使用二分查找…

数栈+AI:数栈V6.2创新发布,让数据开发更智能

近日&#xff0c;以“DataAI&#xff0c;构建新质生产力”为主题的袋鼠云春季发布会圆满落幕&#xff0c;大会带来了一系列“AI”的数字化产品与最新行业沉淀&#xff0c;旨在将数据与AI紧密结合&#xff0c;打破传统的生产力边界&#xff0c;赋能企业实现更高质量、更高效率的…

C语言指针+-整数、指针-指针、指针关系运算、指针和数组、二级指针、指针数组

文章目录 前言一、指针 - 整数二、指针 - 指针三、指针的关系运算四、指针和数组五、二级指针六、指针数组指针数组可以将几个一维数组模拟成二维数组 总结 前言 C语言指针整数、指针-指针、指针关系运算、指针和数组、二级指针、指针数组等介绍&#xff0c;还包括指针数组将几…