数据结构之树结构(下)

各种各样的大树

平衡二叉树 (AVL树)

普通二叉树存在的问题

  1. 左子树全部为空,从形式上看,更像一个单链表

  2. 插入速度没有影响

  3. 查询速度明显降低(因为需要依次比较),不能发挥BST的优势,因为每次还需要比较左子树,其查询速度比单链表还慢

  4. 解决方案-平衡二叉树(AVL)

基本介绍

  • 平衡二叉树也叫平衡二叉搜索树(Self-balancing binary search tree)又被称为AVL树,可以保证查询效率较高。
  • 具有以下特点:它是一棵空树或它的左右两个子树的高度差的绝对值不超过1
    并且左右两个子树都是一棵平衡二叉树。平衡二叉树的常用实现方法有红黑树、
    AVL、替罪羊树、Treap、伸展树等,

应用案例

判断树的高度

// 返回左子树的高度
public int leftHeight() {if (left == null) {return 0;}return left.height();
}// 返回右子树的高度
public int rightHeight() {if (right == null) {return 0;}return right.height();
}// 返回当前结点的高度,以该结点为根结点的树的高度
public int height() {return Math.max(left == null ? 0 : left.height(), right == null ? 0 : right.height()) + 1;
}
单旋转(左旋转)

左旋转步骤

  • 前提条件: r i g h t H e i g h t ( ) − l e f t H e i g h t ( ) > 1 rightHeight() - leftHeight() > 1 rightHeight()leftHeight()>1
  1. 创建一个新的节点newNode(以root 这个值创建),创建一个新的节点,值等于当前根节点的值,把新节点的左子树设置了当前节点的左子树
    // newNode.left = left
  2. 把新节点的右子树设置为当前节点的右子树的左子树
    // newNode.right =right.left;
  3. 把当前节点的值换为右子节点的值
    // value=right.value;
  4. 把当前节点的右子树设置成当前结点的右子树的右子树
    // right=right.right;
  5. 把当前节点的左子树设置为新节点
    // left=newLeft;
// 左旋转
private void leftRotate() {// 1. 创建一个新的节点newNode(以root 这个值创建)// 创建一个新的节点,值等于当前根节点的值,把新节点的左子树设置了当前节点的左子树Node newNode = new Node(value);newNode.left = left;// 2.把新节点的右子树设置为当前节点的右子树的左子树newNode.right = right.left;// 3.把当前节点的值换为右子节点的值value = right.value;// 4.把当前节点的右子树设置成当前结点的右子树的右子树right = right.right;// 5.把当前节点的左子树设置为新节点left = newNode;
}

单旋转(右旋转)

步骤

  1. 创建一个新的节点newNode(以root 这个值创建),创健一个新的节点,值等于当前根节点的值
  2. 把新节点的右子树设置了当前节点的右子树
    // newNode.right right
  3. 把新节点的左子树设置为当前节点的左子树的右子树
    // newNode.left =left.right;
  4. 把当前节点的值换为左子节点的值
    // value=left.value;
  5. 把当前节点的左子树设置成左子树的左子树
    // left=left.left;
  6. 把当前节点的右子树设置为新节点
    // right=newLeft;
// 右旋转
public void rightRotate() {// 1.创建一个新的节点newNode(以root 这个值创建)// 创健一个新的节点,值等于当前根节点的值Node newNode = new Node(value);// 2.把新节点的右子树设置了当前节点的右子树newNode.right = right;// 3.把新节点的左子树设置为当前节点的左子树的右子树newNode.left = left.right;// 4.把当前节点的值换为左子节点的值value = left.value;// 5.把当前节点的左子树设置成左子树的左子树left = left.left;// 6.把当前节点的右子树设置为新节点right = newNode;
}

双旋转

在单旋转中 (即一次旋转) 就可以将非平衡二叉树转成平衡二叉树,但是在某些情况下,单旋转不能完成平衡二叉树的转换

情况1️⃣

在这里插入图片描述

实现步骤

  1. 当右旋转的条件时

    • 如果它的左子树的右子树高度大于它的左子树的右子树的高度
    • 先对当前这个结点的左节点进行左旋转
    • 再对当前结点进行右旋转
  2. 当左旋转时

    • 如果它的右子树的左子树高度大于它的右子树的右子树的高度
    • 先对当前这个结点的右结点进行右旋转
    • 再对当前结点进行左旋转
/*** 添加在Node类的添加结点的方法里边* 每添加一个结点就判断一次
*/
// 单旋转
// 当添加完一个结点后,如果:(右子树的高度-左子树的高度) > 1,左旋转
if (rightHeight() - leftHeight() > 1) {// 如果它的右子树的左子树高度大于它的右子树的右子树的高度if (right != null && right.rightHeight() > right.leftHeight()) {// 先对当前这个结点的右结点进行右旋转rightRotate();}leftRotate();  // 左旋转return; // 一定要返回,否则会出现其他问题,提高VM处理速度
}
// 当添加完一个结点后,如果:(左子树的高度-右子树的高度) > 1,右旋转
if (leftHeight() - rightHeight() > 1) {// 如果它的左子树的右子树高度大于它的左子树的右子树的高度if (left != null && left.rightHeight() > left.leftHeight()) {// 先对当前这个结点的左节点进行左旋转left.leftRotate();}rightRotate();  // 右旋转
}

总代码
package com.xiaolu.avl;/*** @author 林小鹿* @version 1.0*/
public class AvlTreeDemo {public static void main(String[] args) {
//        int[] arr = {4, 3, 6, 5, 7, 8};
//        int[] arr = {10, 12, 8, 9, 7, 6};int[] arr = {10, 11, 7, 6, 8, 9};AVLTree avlTree = new AVLTree();// 添加结点for (int i = 0; i < arr.length; i++) {avlTree.add(new Node(arr[i]));}// 中序遍历avlTree.infixOrder();System.out.println("平衡处理 (左旋转)~~~");System.out.println("树的高度:" + avlTree.getRoot().height()); // 4System.out.println("树的左子树高度:" + avlTree.getRoot().leftHeight()); // 1System.out.println("树的右子树高度:" + avlTree.getRoot().rightHeight()); // 3System.out.println("当前根结点 = " + avlTree.getRoot());}
}// 创建AVLTree
class AVLTree {private Node root;public Node getRoot() {return root;}// 查找要删除的结点public Node search(int value) {if (root == null) {return null;} else {return root.search(value);}}// 查找要删除的父结点public Node searchParent(int value) {if (root == null) {return null;} else {return root.searchParent(value);}}/*** 删除node 为根结点的二叉排序树的最小结点的值的同时返回以node 为根结点的二叉排序树的最小结点** @param node 传入的结点 (当做二叉排序树的根结点)* @return 返回以node 为根结点的二叉排序树的最小结点的值*/public int delRightTreeMin(Node node) {Node target = node;// 循环的查找左结点,就会找到最小值while (target.left != null) {target = target.left;}// 这是target就指向了最小结点// 删除最小结点delNode(target.value);return target.value;}// 删除结点public void delNode(int value) {if (root == null) {return;} else {// 找到需要删除的结点 targetNodeNode targetNode = search(value);// 如果没有找到要删除的结点if (targetNode == null) {return;}// 如果这颗二叉排序树只有一个结点 (即本身就是父节点)if (root.left == null && root.right == null) {root = null;return;}// 查询父节点Node 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) {// 删除有两颗子树的结点int minVal = delRightTreeMin(targetNode.right);targetNode.value = minVal;} else {// 删除只有一颗子树的结点// 如果要删除的结点有左子结点if (targetNode.left != null) {if (parent != null) {// 如果targetNode 是parent 的左子结点if (parent.left.value == value) {parent.left = targetNode.left;} else {// 如果targetNode 是parent 的右子结点parent.right = targetNode.left;}} else {root = targetNode.left;}} else {if (parent != null) {// 如果要删除的结点有右子结点if (parent.left.value == value) {// 如果targetNode 是parent 的左子结点parent.left = targetNode.right;} else {// 如果targetNode 是parent 的右子结点parent.right = targetNode.right;}} else {root = targetNode.right;}}}}}// 添加结点public void add(Node node) {if (root == null) {// 如果root为空则直接让root指向noderoot = node;} else {root.add(node);}}// 中序遍历public void infixOrder() {if (root == null) {System.out.println("二叉树为空,无法遍历");} else {root.infixOrder();}}
}// 创建Node结点
class Node {int value;Node left;Node right;public Node(int value) {this.value = value;}// 返回左子树的高度public int leftHeight() {if (left == null) {return 0;}return left.height();}// 返回右子树的高度public int rightHeight() {if (right == null) {return 0;}return right.height();}// 返回当前结点的高度,以该结点为根结点的树的高度public int height() {return Math.max(left == null ? 0 : left.height(), right == null ? 0 : right.height()) + 1;}// 左旋转private void leftRotate() {// 1. 创建一个新的节点newNode(以root 这个值创建)// 创建一个新的节点,值等于当前根节点的值,把新节点的左子树设置了当前节点的左子树Node newNode = new Node(value);newNode.left = left;// 2.把新节点的右子树设置为当前节点的右子树的左子树newNode.right = right.left;// 3.把当前节点的值换为右子节点的值value = right.value;// 4.把当前节点的右子树设置成当前结点的右子树的右子树right = right.right;// 5.把当前节点的左子树设置为新节点left = newNode;}// 右旋转public void rightRotate() {// 1.创建一个新的节点newNode(以root 这个值创建)// 创健一个新的节点,值等于当前根节点的值Node newNode = new Node(value);// 2.把新节点的右子树设置了当前节点的右子树newNode.right = right;// 3.把新节点的左子树设置为当前节点的左子树的右子树newNode.left = left.right;// 4.把当前节点的值换为左子节点的值value = left.value;// 5.把当前节点的左子树设置成左子树的左子树left = left.left;// 6.把当前节点的右子树设置为新节点right = newNode;}/*** 查找要删除的结点** @param value 待删除的结点的值* @return 如果找到则返回该结点,否则返回空*/public Node 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 返回的是要删除的结点的父结点,如果没有就返回null*/public Node 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; // 没有找到父节点}}}// 按二叉排序树的形式递归添加结点public void add(Node node) {if (node == null) {return;}// 判断传入的结点的值与当前子树的根结点的值关系if (node.value < this.value) {if (this.left == null) {this.left = node;} else {// 递归得向左子树添加 nodethis.left.add(node);}} else { // 添加的结点的值大于当前结点的值if (this.right == null) {this.right = node;} else {// 递归得向右子树添加 nodethis.right.add(node);}}// 单旋转// 当添加完一个结点后,如果:(右子树的高度-左子树的高度) > 1,左旋转if (rightHeight() - leftHeight() > 1) {// 如果它的右子树的左子树高度大于它的右子树的右子树的高度if (right != null && right.leftHeight() > right.rightHeight()) {// 先对当前这个结点的右结点进行右旋转rightRotate();}leftRotate();  // 左旋转return; // 一定要返回,否则会出现其他问题,提高VM处理速度}// 当添加完一个结点后,如果:(左子树的高度-右子树的高度) > 1,右旋转if (leftHeight() - rightHeight() > 1) {// 如果它的左子树的右子树高度大于它的左子树的右子树的高度if (left != null && left.rightHeight() > left.leftHeight()) {// 先对当前这个结点的左节点进行左旋转left.leftRotate();}rightRotate();  // 右旋转}}// 中序遍历public void infixOrder() {if (this.left != null) {this.left.infixOrder();}System.out.println(this);if (this.right != null) {this.right.infixOrder();}}@Overridepublic String toString() {return "Node[" +"value=" + value +']';}
}

多路查找树

在这里插入图片描述

二叉树存在的问题

  • 二叉树需要加载到内存的,如果二叉树的节点少,没有什么问题,但是如果二叉树的节点很多(比如1亿),就存在如下问题

  • 问题1:在构建二叉树时,需要多次进行 I / O I/O I/O 操作海量数据存在数据库或文件中),节点海量,构建二叉树时,速度有影响

  • 问题2:节点海量,也会造成二叉树的高度很大,会降低操作速度

多叉树

在二叉树中,每个节点有数据项,最多有两个子节点,如果允许每个节点可以有更多的数据项和更多的子节点,就是多叉树(multiway tree)

多叉树通过重新组织节点,减少树的高度,能对二叉树进行优化

2-3树

2-3树是最简单的B-树结构,其他像2-3-4树同理

特点

  • 2-3树的所有叶子节点都在同一层.(只要是B树都满足这个条件)
  • 有两个子节点的节点叫二节点,二节点要么没有子节点,要么有两个子节点
  • 有三个子节点的节点叫三节点,三节点要么没有子节点,要么有三个子节点.
  • 2-3树是由二节点和三节点构成的树。

插入规则:

1)2-3树的所有叶子节点都在同一层.(只要是B树都满足这个条件)
2)有两个子节点的节点叫二节点,二节点要么没有子节点,要么有两个子节点

3)有三个子节点的节点叫三节点,三节点要么没有子节点,要么有三个子节点
4)当按照规则插入一个数到某个节点时,不能满足上面三个要求,就需要拆,先向上拆,如果上层满,则拆本层,拆后仍然需要满足上面3个条件。
5)对于三节点的子树的值大小仍然遵守(BST 二叉排序树)的规则


B树

B-tree,B即Balanced

基本介绍

  • B树的阶:节点的最多子节点个数。比如2-3树的阶是3,2-3-4树的阶是4
  • B树的搜索,从根结点开始,对结点内的关键字(有序)序列进行二分查找,如果命中侧结束,否测进入查询关键字所属范围的儿子结点;重复,直到所对应的儿子指针为空,或已经是叶子结点
  • 关键字集合分布在整颗树中,即叶子节点和非叶子节点都存放数据
  • 搜索有可能在非叶子结点结束
  • 其搜索性能等价于在关键字全集内做一次二分查找

在这里插入图片描述


B+树

B+树是B树的变体,也是一种多路搜索树

基本介绍

  • B+树的搜索与B树也基本相同,区别是B+树只有达到叶子结点才命中(B树可以在非叶子结点命中),其性能也等价于在关键字全集做一次二分查找
  • 所有关键字都出现在叶子结点的链表中(即数据只能在叶子节点【也叫稠密索引】),且链表中的关键字数据恰好是有序的。
  • 不可能在非叶子结点命中
  • 非叶子结点相当于是叶子结点的索引(稀疏索引)叶子结点相当于是存储(关键字)数据的数据层
  • 更适合文件索引系统
  • B树和B+时各有自己的应用场景,不能说B+完全比B树好,反之亦然

在这里插入图片描述


B*树

B*树是B+树的变体,在B+树的非根和非叶子结点再增加指向兄弟的指正

基本说明

  • B*树定义了非叶子结点关键字个数至少为(2/3)*M,即块的最低使用率为2/3,而B+树的块的最低使用率为B+树的1/2。
  • 从第1个特点我们可以看出,B*树分配新结点的概率比B+树要低,空间使用率更高

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

javaWeb个人学习04

AOP核心概念: 连接点: JoinPoint, 可以被AOP控制的方法 通知: Advice 指哪些重复的逻辑&#xff0c;也就是共性功能(最终体现为一个方法) 切入点: PointCut, 匹配连接点的条件&#xff0c;通知仅会在切入点方法执行时被应用 目标对象: Target, 通知所应用的对象 通知类…

docker基线安全修复和容器逃逸修复

一、docker安全基线存在的问题和修复建议 1、将容器的根文件系统挂载为只读 修复建议&#xff1a; 添加“ --read-only”标志&#xff0c;以允许将容器的根文件系统挂载为只读。 可以将其与卷结合使用&#xff0c;以强制容器的过程仅写入要保留的位置。 可以使用命令&#x…

航拍无人机技术,航拍无人机方案详解,无人机摄影技术

航拍无人机是利用遥控技术和摄像设备&#xff0c;在空中进行拍摄和录像的无人机。这种无人机通常具有高清摄像设备、图像传输设备、GPS定位系统、智能控制系统等&#xff0c;可以轻松实现各种拍摄角度和高度&#xff0c;广泛应用于影视制作、旅游景区航拍、城市规划、环保监测等…

【数据结构与算法】回溯法解题20240301

这里写目录标题 一、78. 子集1、nums [1,2,3]为例把求子集抽象为树型结构2、回溯三部曲 二、90. 子集 II1、本题搜索的过程抽象成树形结构如下&#xff1a; 三、39. 组合总和1、回溯三部曲2、剪枝优化 四、LCR 082. 组合总和 II1、思路2、树形结构如图所示&#xff1a;3、回溯…

用vivado创建一个赛灵思AXI的IP核

一、新建一个管理IP的任务 二、设置板子&#xff0c;verilog语言和文件位置 三、创建新的IP核 添加一个axi-full的master接口和axi-full的slave接口 四、查看赛灵思AXI代码 第一个是axi的master接口代码&#xff0c;下面的是axi的slave接口代码 五、打包IP核以供后续使用 六、…

出现 ‘vue‘ 不是内部或外部命令,也不是可运行的程序 或批处理文件的解决方法(图文界面)

目录 前言1. 问题所示2. 原理分析3. 解决方法前言 由于Java转全栈,对此前端的细节点都比他人更加注意,所以此处记录更有用的信息!(小白都能看懂) 1. 问题所示 出现如下问题: F:\vue_project>vue -version vue 不是内部或外部命令,也不是可运行的程序 或批处理文件…

基于Python的电商评论数据采集与分析|电商API接口数据采集

引言 在电商竞争日益激烈的情况下&#xff0c;商家既要提高产品质量&#xff0c;又要洞悉客户的想法和需求&#xff0c;关注客户购买商品后的评论&#xff0c;而第三方商家获取商品评价主要依赖于人工收集&#xff0c;不但效率低&#xff0c;而且准确度得不到保障。通过使用Py…

鸿蒙 渲染控制

前提&#xff1a;基于官网3.1/4.0文档。参考官网文档 基于Android开发体系来进行比较和思考。&#xff08;或有偏颇&#xff0c;自行斟酌&#xff09; 1.概念 ArkUI通过自定义组件的build()函数和builder装饰器中的声明式UI描述语句构建相应的UI。在声明式描述语句中开发者除了…

Ps:绘画对称功能

Photoshop 中的绘画对称 Paint Symmetry功能允许用户在画布上创建对称的绘画和设计&#xff0c;极大地提高了创作的效率和准确性&#xff0c;尤其适合于制作复杂的对称图形和图案。 可在使用画笔工具、铅笔工具或橡皮擦工具时启用“绘画对称"功能。 提示&#xff1a; 绘画…

Cocos游戏开发中的金币落袋效果

引言 Cocos游戏开发中的金币落袋效果 大家好,不知道大家有没有被游戏中的一些小细节打动或吸引。 往往游戏就是通过一些与众不同的细节,去留住玩家。 金币落袋效果正是如此,它比普通的数值变化来得更加形象,给予玩家成就感和满足感。 本文重点给大家介绍一下如何在Coc…

Opencv基本操作 (上)

目录 图像基本操作 阈值与平滑处理 图像阈值 图像平滑处理 图像形态学操作 图像梯度计算 Sobel 算子 Canny 边缘检测 图像金字塔与轮廓检测 图像轮廓 接口定义 轮廓绘制 轮廓特征与相似 模板匹配 傅里叶变换 傅里叶变换的作用 滤波 图像基本操作 读取图像&…

【Maven】Maven 基础教程(二):Maven 的使用

《Maven 基础教程》系列&#xff0c;包含以下 2 篇文章&#xff1a; Maven 基础教程&#xff08;一&#xff09;&#xff1a;基础介绍、开发环境配置Maven 基础教程&#xff08;二&#xff09;&#xff1a;Maven 的使用 &#x1f60a; 如果您觉得这篇文章有用 ✔️ 的话&#…

Qt中关于信号与槽函数的思考

信号与槽函数的思考 以pushbutton控件为例&#xff0c;在主界面上放置一个pushbutton控件&#xff0c;点击右键选择关联槽函数&#xff0c;关联一个click函数&#xff0c;如下图所示&#xff1a; 在该函数中&#xff0c;实现了一个点击pushbutton按钮后&#xff0c;弹出一个窗…

go并发模式之----使用时顺序模式

常见模式之二&#xff1a;使用时顺序模式 定义 顾名思义&#xff0c;起初goroutine不管是怎么个先后顺序&#xff0c;等到要使用的时候&#xff0c;需要按照一定的顺序来&#xff0c;也被称为未来使用模式 使用场景 每个goroutine函数都比较独立&#xff0c;不可通过参数循环…

DOM 获取父子节点

DOM 是以树状结构排列的&#xff0c;所以父子关系是相对的&#xff0c;当li为我们的目标节点的时候&#xff0c;ul为其父节点&#xff0c;其他li为它的兄弟节点&#xff0c;li里面包含的标签为子节点&#xff0c;以此类推。 那我们如何找父节点&#xff1f; 元素.parentNode&am…

【计算机网络——应用层】http协议

文章目录 1. http协议1.1 http协议简介1.2 url组成1.3 urlencode与urldecode 2. http协议的格式2.1 http协议的格式2.2 一些细节问题 3. http的方法、状态码和常见响应报头3.1 http请求方法3.2 http状态码3.3 http常见的响应报头属性 4. 一个非常简单的http协议服务端5. http长…

【X806开发板试用】文章一 ubuntu开发环境搭建

一、环境配置 官方链接&#xff1a; 环境配置 1.安装必要的库和软件 sudo apt-get install build-essential gcc g make zlib* libffi-dev e2fsprogs pkg-config flex bison perl bc openssl libssl-dev libelf-dev libc6-dev-amd64 binutils binutils-dev libdwarf-dev u-b…

pix2pix-zero

pix2pix-zero&#xff1a;零样本图像到图像转换 论文介绍 Zero-shot Image-to-Image Translation 关注微信公众号: DeepGoAI 项目地址&#xff1a;https://github.com/pix2pixzero/pix2pix-zero 论文地址&#xff1a;https://arxiv.org/abs/2302.03027 本文介绍了一种名为…

基于SpringBoot多模块项目引入其他模块时@Autowired无法注入

基于SpringBoot多模块项目引入其他模块时Autowired无法注入 一、问题描述1、解决方案 一、问题描述 启动Spring Boot项目时报 Could not autowire. No beans of ‘xxxxxxxx’ type found. 没有找到bean的实例&#xff0c;即spring没有实例化对象&#xff0c;也就无法根据配置文…

【LeetCode-中等】209.长度最小的子数组-双指针/滑动窗口

力扣题目链接 1. 暴力解法 这道题的暴力解法是两层嵌套for循环&#xff0c;第一层循环从 i 0 开始遍历至数组末尾&#xff0c;第二层循环从 j i 开始遍历至找到总和大于等于 target 的连续子数组&#xff0c;并将该连续子数组的长度与之前找到的子数组长度相比较&#xff0…