树的基本定义

news/2024/11/17 9:28:46/

树的基本术语

一个节点包括一下内容

  1. 父节点的地址值
  2. 左节点的地址值
  3. 右节点的地址值

如果没有父节点或者没有左右节点,那么这些节点对应的位置是 null
常见术语

  1. 节点—树中的元素常称为节点

  2. 边—根和它的子树根(如果存在)之间形成边的边可到达另一个结点,则称这两个结点间存在一条路径。

  3. 双亲—若一个结点有子树,那么该结点称为子树根的双亲

  4. 孩子—子树的根是该结点的孩子

  5. 兄弟—有相同双亲的结点互为兄弟

  6. 后裔—一个结点的所有子树上的任何结点都是该结点的后裔

  7. 祖先—从根结点到某个结点路径上的所有结点都是该结点的祖先

  8. 度—一个结点拥有的子树数量称为该结点的度。度为 0 的结点称为叶子结点,树中结点的最大的度称为树的度

  9. 层次—一般将根节点的层次定义为 1, 其余节点的层次等于其双亲结点的层次加 1, 书中节点的最大层次称为该树的高度

  10. 无序树, 有序树, 如果一棵树中各结点的子树的次序不重要,可以交换位置,这样的树称为无序树。如果将树中结点的各棵子树看成是从左到右有次序的,则称该树为有序树。从左到右,可分别称这些子树为第一子树、第二子树等。

  11. 森林—森林是树的有限集合

左子树

  • 对于任意一个节点 n,它的左子树是从根节点到 n 的路径上所有节点的集合,包括 n 自己。

右子树

  • 对于任意一个节点 n,它的右子树是从根节点到 n 的路径上所有节点的集合,不包括 n 自己。

二叉查找树
左节点的值比父亲节点的值小
右节点的值比父亲节点的值大
我们经常使用这个二叉查找树来进行查找节点

二叉树的遍历方式

三种递归遍历

  1. 我们定义了二叉树的四种遍历运算,即先序遍历、中序通历、后序遍历和层次遍历。其中先序遍历、中序遍历和后序遍历的设计思想与二叉树的递归定义密切相关

层次遍历

  1. 层次遍历是利用二叉树中各结点所在的层次,按照从上到下、从左到右的顺序访问二叉树中的每一个结点。

先序, 中序, 后序遍历

  1. 对于这三种遍历方法而言, 遍历的逻辑是没有什么区别的, 唯一的区别在于, 什么时候输出

  2. 函数的递归调用, 就是函数的自身调用, 执行完之后会返回上一个被嵌套的函数, 去执行, 未执行的表达式

  3. 假设 L、V 和 R 分别代表遍历左子树、访问根结点和遍历右子树这三个操作,那么就可以得到六种遍历次序,分别是 VLR、LVR、LRV、VRL、RVL 和 RLV。

  4. 即先访问根结点的先序遍历 VLR、中间访问根结点的中序遍历 LVR 和最后访问根结点的后序遍历 LRV

三种遍历算法的总结

  1. 先序遍历算法读取的时候输出是在第一个,也就是说每次执行函数都会有输出,其是先从根节点开始的,然后遍历左子树,和右子树,所以它的结果一般是,从根节点开始输出,然后不断输出根节点的左子树内容,然后是到最下面的左子树结点的时候,判断有没有左右子树,没有的话,判断上一个有没有右子树,有的话先输出右子树,然后在继续判断有没有左子树,,,,

  2. 中序遍历算法输出在遍历左子树之后,到最下面一个左子树才开始输出,然后是判断其有没有右子树,没有的话,返回上一步,此时上一步的遍历左子树算法结束,然后输出结点,然后继续遍历右子树,如果是先序遍历的话,此时这个右子树会输出的,但是中序遍历却不会输出,而是继续判断这个右子树有没有左子树之后才会输出。这就是区别

  3. 后序遍历,就是先到最下方的一个左子树,然后是返回上一个节点,此时先序和中序,都会输出上面的结点,但是后序不会,后序的话,是只有子树输出完了,才会输出双亲结点

前序遍历从跟节点开始左子节点,右子节点的顺序遍历
当前,左,右的遍历方式
中序遍历我们先遍历最左边的节点(即最左边的叶子节点),等这个节点遍历完之后我们就往上一层,如果有右节点,那么先观察它有没有左节点,如果有的话那么我们就遍历左节点
记住关键点,一切都是以左节点为先
后序遍历
左,右,当前节点
层次遍历
一层一层的遍历
java 中常见的二叉树类型其实和 C 语言是类似的,并不是通过其特有的 LinkHashSet 还是 LinkArraylist,亦或者是 LinkHashMap, 本质上还是通过类来进行构建的

class TreeNode {int val;TreeNode left;TreeNode right;public TreeNode(int val) {this.val = val;this.left = null;this.right = null;}
}public class BinaryTree {private TreeNode root;public BinaryTree() {this.root = null;}// 添加节点到二叉树public void addNode(int val) {root = addNodeRecursive(root, val);}// 递归方式添加节点到二叉树private TreeNode addNodeRecursive(TreeNode current, int val) {if (current == null) {return new TreeNode(val);}if (val < current.val) {current.left = addNodeRecursive(current.left, val);} else if (val > current.val) {current.right = addNodeRecursive(current.right, val);}return current;}// 遍历二叉树(示例为中序遍历)public void traverse() {traverseInOrder(root);}// 中序遍历二叉树private void traverseInOrder(TreeNode node) {if (node != null) {traverseInOrder(node.left);System.out.print(node.val + " ");traverseInOrder(node.right);}}public static void main(String[] args) {BinaryTree tree = new BinaryTree();// 添加节点tree.addNode(50);tree.addNode(30);tree.addNode(20);tree.addNode(40);tree.addNode(70);tree.addNode(60);tree.addNode(80);// 遍历二叉树tree.traverse();}
}

平衡二叉树

平衡二叉树的特点
二叉树左右两个子树的高度差不超过 1
任意节点的左右两个子树都是一颗平衡二叉树

平衡二叉树旋转
旋转触发时机
当添加一个节点之后, 该树不再是一颗平衡二叉树
左旋
就是将根节点的右侧往左拉, 原先的右子节点变成新的父节点, 并把多余的左子节点出让, 给已经降级的根节点当右子节点
从添加的节点开始,不断的往父节点中找不平衡的点
image.png
外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
image.png

平衡右旋
右旋
就是将根节点的左侧往右拉, 左子节点变成了新的父节点, 并把多余的右子节点出让, 给已经降级根节点当左子节点
image.png

image.png

左右
左右: 当根节点左子树的右子树有节点插入, 导致二叉树不平衡
如何旋转: 先在左子树对应的节点位置进行左旋, 在对整体进行右旋
外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

我们并不能通过一次旋转来实现这个二叉树的变换

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

红黑树

image.png

image.png

简单路径其实综合来说就是,不能回头
我们通过第五条规则来设定这个红色节点的数量

添加节点的规则

默认添加的节点是红色的,这样的话效率高

image.png

叔叔节点就是父节点的兄弟节点
根据这个操作来说,是不难的

红黑树是一种增删改查数据性能相对都较好的结构


http://www.ppmy.cn/news/1094328.html

相关文章

异步编程 - 07 基于JDK中的Future实现异步编程(下)_当Stream遇见CompletableFuture

文章目录 JDK8 StreamStream遇见CompletableFuture小结 JDK8 Stream JDK8中提供了流式对数据进行处理的功能&#xff0c;它的出现允许我们以声明式方式对数据集合进行处理。所谓声明式是相对于我们平时所用的命令式编程来说的&#xff0c;使用声明式编程会让我们对业务的表达更…

MySQL聚簇索引与非聚簇索引

分析&回答 当数据库一条记录里包含多个字段时&#xff0c;一棵B树就只能存储主键&#xff0c;如果检索的是非主键字段&#xff0c;则主键索引失去作用&#xff0c;变成顺序查找了。这时应该在第二个要检索的列上建立第二套索引。这个索引由独立的B树来组织。有两种常见的方…

rhcsa学习1基本命令(软硬链接,获取帮助等)

pwd 显示该shell的当前工作目录的完整路径名 ls 列出指定目录的目录内容 cd 更改shell的当前工作目录 没有什么特殊情况命令小写 ~所在位置&#xff1a;当前工作目录 #当前用户为管理员root $当前是普通用户 命令由三个基本部分组成&#xff1a; 命令、选项、参数 如下 -a --a…

Java LinkedList

简介 链表&#xff08;Linked list&#xff09;是一种常见的基础数据结构&#xff0c;是一种线性表&#xff0c;但是并不会按线性的顺序存储数据&#xff0c;而是在每一个节点里存到下一个节点的地址。 链表可分为单向链表和双向链表。 在Java程序设计语言中&#xff0c;所有…

Pytorch实现鸟类品种分类识别(含训练代码和鸟类数据集)

Pytorch实现鸟类品种分类识别(含训练代码和鸟类数据集) 目录 Pytorch实现鸟类品种分类识别(含训练代码和鸟类数据集) 1. 前言 2. 鸟类数据集 &#xff08;1&#xff09;Bird-Dataset26 &#xff08;2&#xff09;自定义数据集 3. 鸟类分类识别模型训练 &#xff08;1&a…

算法设计与分析 | 页码统计

题目&#xff1a; 一本书的页码从自然数1开始顺序编码直到自然数n。书的页码按照通常的习惯编排&#xff0c;每个页码都不含多余的前导数字0。例如第6页用6表示而不是06或006。数字统计问题要求对给定书的总页码&#xff0c;计算出书的全部页码中分别用到多少次数字0,1,2,3,...…

【嵌入式数据库之sqlite3】

目录 一.数据库基本概念&#xff08;理解&#xff09; 1.数据 2.数据库 二.常用的数据的数据库&#xff08;了解&#xff09; 1.大型数据库 2.中型数据库 3.小型数据库 三.基于嵌入式的数据库&#xff08;了解&#xff09; 四.SQLite基础&#xff08;了解&#xff09;…

SQL数据类型

数据类型 整数类型浮点数类型定点数类型日期类型时间类型字符串类型二进制类型… 整数类型 MySQL 中 整数类型有5种 整数类型字节数无符号取值范围有符号取值范围tinyint10~255-128-127smallint20~65 535-32 768-32 768mediumint30~16 777 215-8 388 608-8 388 608int40~4 …