JavaDS —— 二叉树

树的基本概念

树是一种非线性的数据结构,它是由n(n>=0)个有限结点组成一个具有层次关系的集合。把它叫做树是因为它看
起来像一棵倒挂的树,也就是说它是根朝上,而叶朝下的。

树形结构中,子树之间不能有交集,否则就不是树形结构

在这里插入图片描述


在这里插入图片描述

结点的度:一个结点包含的子树的个数称为结点的度。 例如上图 :A的度为 6,D的度为1,B的度为0

树的度:一颗树中,所有结点度的最大值称为树的度。 例如上图:A 的度为6是所有结点的度的最大值,所以树的度为 6

叶子结点或者终端结点: 度为0的结点称为叶子结点或者终端结点。 例如上图: B,C,H,I,P,Q,K,L,M,N是叶子结点

双亲结点或者父结点: 若一个结点包含子节点,则这个结点就是子节点的双亲结点。 例如上图: A是B的双亲结点,D是H的双亲结点

孩子结点或子结点:一个结点所在的子树的根节点称为该结点的子节点。 例如:B是A孩子结点,P和Q是J的孩子结点

根结点:一个树中,没有双亲结点的结点。 A是整棵树的根结点,F是树(F,K,L,M构成的树)的根节点。

结点的层次:从根结点开始定义,根为第一层,根的子结点为第二层,以此类推

树的高度或深度: 树中结点的最大层次,根结点定义为1,例如上图所示整棵树的高度为4

非终端结点或分支结点:度不为0的结点,例如:D,E,F,G等等结点是分支结点

兄弟结点:具有相同的双亲结点的结点互称为兴地结点。 例如B,C,D,E,F,G的双亲结点都是A,那么它们都是对方的兴地结点

堂兄弟结点:双亲在同一层的结点互为堂兄弟。 例如上图:H,I,K互为堂兄弟结点

结点的祖先:从根到该系欸但所经分支的所有结点。 A是所有结点的祖先。

子孙:以某结点为根的子树中任一结点都称为该结点的子孙。 例如上图:A是所有结点的祖先

森林:由m (m>=0) 棵互不相交的树组成的集合称为森林。

树的表示形式

树结构相对线性表就比较复杂了,要存储表示起来就比较麻烦了,实际中树有很多种表示方式,如:双亲表示法,孩子表示法、孩子双亲表示法、孩子兄弟表示法等等。这里就简单介绍其中最常用的孩子兄弟表示法。

在这里插入图片描述

二叉树的概念与性质

二叉树中所有的结点的度都小于等于2,并且可以分为左子树和右子树,是一个有序树
在这里插入图片描述

二叉树的几种形态:
在这里插入图片描述
在这里插入图片描述

满二叉树

一棵二叉树,如果每层的结点数都达到最大值,则这棵二叉树就是满二叉树。也就是说,如果一棵二叉树的层数为K,且结点总数是 2^k - 1 ,则它就是满二叉树。
在这里插入图片描述

完全二叉树

完全二叉树是效率很高的数据结构,完全二叉树是由满二叉树而引出来的。对于深度为K的,有n个结点的二叉树,当且仅当其每一个结点都与深度为K的满二叉树中编号从0至n-1的结点一一对应时称之为完全二叉树。通俗一点来说,完全二叉树就是只有最后一层可能不是满的,并且最后一层的结点排列是从左到右依次排列的。 要注意的是满二叉树是一种特殊的完全二叉树
在这里插入图片描述

性质

1.若规定根节点的层数为1,则一颗非空二叉树的第 i 层最多有 2 ^ ( i - 1 ) 个结点(i > 0)

证明:下面是一颗满二叉树。
在这里插入图片描述
第一层有 1 个结点
第二层有 2 个结点
第三层有 4 个结点
。。。。。。

可以看出这是一个等比数列,使用等比数列求某一项的公式可得:第 i 层的结点数最多为 2 ^ ( i - 1 )


2.若规定只有根结点的二叉树的深度为1,则深度为 k 的二叉树结点总数最多为 2 ^ k - 1 (k > 0)

证明:
还是拿上面的满二叉树举例,现在是求深度为 k 的最多的结点总数是多少?这里很简单直接使用等比数列求和公式,2 ^ 0 + 2 ^ 1 + 2 ^ 2 + 2 ^ 3 + … = 2 ^ 0 * (1 - 2 ^ k) / (1 - 2 ) = 2 ^ k - 1


3.对于任意一颗二叉树,如果其叶子结点个数为 n0,度为2的分支结点个数为 n2 ,则 n0 = n2 + 1

证明:
二叉树的结点总数等于 度为 0 的叶子节点 加 度为 1 的根节点 加 度为 2 的分支节点,设节点总数为 N,度为 0 的叶子节点总数为 n0 , 度为1的根节点的数量为 n1 ,度为2的分支节点数量为 n2 , N = n0 + n1 + n2
在二叉树中 N 个节点就会产生 N - 1 条边,因为每一个节点上面都会有一条边连接着,除了根节点,所以要减一。
除了上面这种即使边的方式,还能这样理解,度为1 的节点 会向下延伸一条边,度为 2 的节点 会向下延伸两条条边,度为 0 的节点 不会向下延伸,则边的总数还能等于 1 * n1 + 2 * n2

联立方程:
N = n0 + n1 + n2
边的总数等于 N - 1
边的总数还能等于 1 * n1 + 2 * n2

即 n0 + n1 + n2 - 1 = 1 * n1 + 2 * n2
n0 = n2 + 1


  1. 具有n 个结点的完全二叉树的深度 k 为 log( n + 1 ) 【以2为底的对数】 结果向上取整

证明:
由性质二可知 深度为 k 的二叉树结点总数 n 最多为 2 ^ k - 1 (k > 0) ,则取对数可得 k = log( n + 1 ) 结果向上取整,因为完全二叉树最后一层可能不是满的,所以你得到的结果可能有小数部分,这时候最后一层也是算的,所以结果要向上取整。


性质5:
对于具有 n 个结点的完全二叉树,如果按照从上到下从左到右的顺序对所有的结点从0开始编号,则对于序号为 i 的结点有:
若 i = 0,i 为根节点的编号,没有双亲结点
若 i > 0 , 双亲结点的序号为 (i - 1) / 2
若 2 * i + 1 < n , 左孩子序号为 2 * 1 + 1,否则没有左孩子
若 2 * i + 2 < n , 右孩子序号为 2 * 1 + 2,否则没有右孩子

二叉树的实现

创建二叉树

public class TreeNode {char val;TreeNode left;TreeNode right;public TreeNode(char val) {this.val = val;}
}

创建方法一:这个方法可以实现自己想要的二叉树,而且写起来比较方便。

    //创建二叉树public TreeNode creatTree() {TreeNode root = new TreeNode('A');TreeNode B = new TreeNode('B');TreeNode C = new TreeNode('C');TreeNode D = new TreeNode('D');TreeNode E = new TreeNode('E');TreeNode F = new TreeNode('F');root.left = B;root.right = C;B.left = D;B.right = E;C.left = F;return root;}

创建方法二:根据前序遍历中序遍历或者后序遍历的结果进行创建,但是有一个前提就是必须提供所有的空树和所有的结点的序列。

下面的创建方法是基于前序遍历的字符串使用的。

    public static int i;public TreeNode creatTree(String str){TreeNode root = null;if(str.charAt(i) == '#') {i++;return null;}root = new TreeNode(str.charAt(i));i++;root.left = creatTree(str);root.right = creatTree(str);return root;}

这里使用了一个外部的变量 i ,这个变量是用来表示此时指向字符串的哪个字符

由于这是一个静态的变量所有所有使用Tree实例化的对象都应该只有一份的 i ,这就意味着如果创建多个树的时候需要重置一下 i 的数值。

前序遍历

前序遍历又可以称为先序遍历,遍历方式是先访问根节点的数值,然后遍历左子树,最后遍历右子树。

    public void preOreder(TreeNode root) {if(root == null) {return;}System.out.println(root.val + " ");preOreder(root.left);preOreder(root.right);}

迭代的方法在 二叉树 OJ (一)的习题文章中,大家可在文末点开链接阅读

中序遍历

中序遍历是先遍历左子树,然后访问根节点的数值,最后遍历右子树

    public void inOreder(TreeNode root) {if(root == null) {return;}inOreder(root.left);System.out.println(root.val + " ");inOreder(root.right);}

迭代的方法在 二叉树OJ(一)的习题文章中,大家可以点开链接阅读

后序遍历

后序遍历是先遍历左子树,然后遍历右子树,最后访问根节点的数值。

    public void postOreder(TreeNode root) {if(root == null) {return;}postOreder(root.left);postOreder(root.right);System.out.println(root.val + " ");}

迭代的方法在 二叉树OJ(一)的习题文章中,大家可以点开链接阅读

如何从遍历序列推导二叉树?

先序遍历能确定根节点,后序遍历也能确定根结点,中序遍历在得知根节点的前提下能推导根节点左右子树序列,所以要想从遍历序列推导出二叉树,就一定要知道中序遍历序列,还要知道先序遍历或者后序遍历序列其中之一即可。


已知某二叉树的前序遍历序列为ABDEC,中序遍历序列为BDEAC,那么它的后序遍历序列是什么?

首先根据前序序列,我们知道根节点为 A,然后根据中序序列找到 A,发现A的左边为 BDE,右边为 C,即获得A的右子树为C,如下图:
在这里插入图片描述
由于前序遍历是根左右,那么 B 就是 A 的左孩子,也是左子树的根节点,然后回到中序遍历序列找到 结点 B ,发现结点 B 左边没有其他结点所以 B 的左子树为空,右子树为 DE:
在这里插入图片描述
继续之前的步骤,从先序遍历得到 D 是根节点,再看中序遍历得到 E 在 B 的右边,所以 E 是 B 的右子树:
在这里插入图片描述
然后根据还原的二叉树写出后序遍历的序列:EDBCA


已知某二叉树的后序遍历序列为EDBCA,中序遍历序列为BDEAC,那么它的前序遍历序列是什么?

根据后序遍历的特点,最后一个结点为根节点即为 A,然后到中序遍历中找到 A,A 的左子树为 BDE,右子树为 C :
在这里插入图片描述

然后后序遍历的倒数第二个结点为 C ,回到中序遍历可得 C 的左子树右子树都为空,继续,后序遍历倒数第三个结点为 B,根据中序遍历序列可得 B 的左子树为空,右子树为 DE:
在这里插入图片描述
重复上诉步骤,最后可得:
在这里插入图片描述
则前序遍历的结果为 ABDEC

这里不演示代码,代码放在 二叉树OJ(一)文章中,大家可以打开链接自行查阅。

层序遍历

层序遍历是从上到下从左到右依次遍历每一层的结点。

这里我们需要队列这个数据结构来存放每一个结点,通过出队打印数值,然后入队这个出了队的结点的左右孩子结点(不为空的结点进行入队),之后就是一直重复出队、入队、出队、入队等等这些操作,通过循环实现,直到所有结点遍历完成。

    public void levelOrder(TreeNode root) {if(root == null) {return;}Queue<TreeNode> queue = new LinkedList<>();queue.offer(root);while(!queue.isEmpty()) {TreeNode tmp = queue.poll();System.out.print(tmp.val + " ");if(tmp.left != null) {queue.offer(tmp.left);}if(tmp.right != null) {queue.offer(tmp.right);}}System.out.println();}

获取结点总数

使用递归先遍历左子树再遍历右子树,当结点不为空的时候 nodes++

    public int nodes;public void size(TreeNode root) {if(root == null) {return;}nodes++;size(root.left);size(root.right);}

除了可以使用遍历思路,还可以采用子问题思路,就是将问题变成结点总数等于左子树的结点总数加右子树的结点总数加根节点。

    //计算结点个数//左子树节点数加右子树节点数加根节点public int size(TreeNode root) {if(root == null) {return 0;}return size(root.left) + size(root.right) + 1;}

获取叶子结点总数

可以采用遍历思路,遍历所有结点,将不为空的结点使用计数器++即可

    public int leafNodes;public void getLeafNodes(TreeNode root) {if(root == null) {return;}if(root.left == null && root.right == null) {leafNodes++;}getLeafNodes(root.left);getLeafNodes(root.right);}

也可以采用子问题,叶子结点数目等于左子树的叶子加右子树的叶子。

    public int getLeafNodes(TreeNode root) {if(root == null) {return 0;}if(root.left == null && root.right == null) {return 1;}return getLeafNodes(root.left) + getLeafNodes(root.right);}

获取树的高度或深度

采用子问题思路,树的高度是左右子树的最大高度,然后加一,因为还有根节点。

    public int getHeight(TreeNode root) {if(root == null) {return 0;}int leftHeight = getHeight(root.left);int rightHeight = getHeight(root.right);return Math.max(leftHeight,rightHeight) + 1;}

获取第 k 层的结点个数

子问题思路:左子树的第 k 层节点数 加 右子树的第 k 层结点数 ,在根节点不为空的前提下,当k 就是1 的时候,那么返回 1 .

    public int getKNodes(TreeNode root,int k) {if(root == null) {return 0;}if(k == 1) {return 1;}return getKNodes(root.left,k-1) + getKNodes(root.right,k-1);}

判断一棵树是不是完全二叉树

在这里插入图片描述

从上图可以得知:如果我们把空节点也入队的话,那么如果是完全二叉树最后存放的结点应该全是 null,否则就不是完全二叉树。

    public boolean isCompleteTree(TreeNode root) {if(root == null) {return true;}Queue<TreeNode> queue = new LinkedList<>();queue.offer(root);while(!queue.isEmpty()) {TreeNode tmp = queue.poll();if(tmp != null) {queue.offer(tmp.left);queue.offer(tmp.right);} else {break;}}while(!queue.isEmpty()) {TreeNode tmp = queue.poll();if(tmp != null) {return false;}}return true;}

平衡二叉树

平衡二叉树 是指该树所有节点的左右子树的深度相差不超过 1

要注意是所有结点的子树!!!
在这里插入图片描述
上图则是一颗平衡二叉树。


在这里插入图片描述
这就不是一颗平衡二叉树。


二叉搜索树

‌二叉搜索树(Binary Search Tree),也称为二叉查找树或二叉排序树,是一种特殊的二叉树。它的定义基于以下性质:

若它的左子树不空,则左子树上所有节点的值都小于根节点的值
若它的右子树不空,则右子树上所有节点的值都大于根节点的值
它的左、右子树也分别为二叉搜索树。
此外,二叉搜索树的一个重要特性是它的中序遍历结果一定是有序的。这意味着在二叉搜索树中,如果按照中序遍历的方式访问所有节点,将得到一个有序的节点值序列。‌

二叉搜索树的这些性质使得它在数据检索、排序等算法中具有高效性,尤其是在需要频繁查找、插入或删除数据的场景中,二叉搜索树的操作效率通常优于其他数据结构。

习题文章链接:
http://t.csdnimg.cn/YYNy7

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

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

相关文章

健康问题查询找搜索引擎还是大模型

随着自然语言处理&#xff08;NLP&#xff09;的最新进展&#xff0c;大型语言模型&#xff08;LLMs&#xff09;已经成为众多信息获取任务中的主要参与者。然而&#xff0c;传统网络搜索引擎&#xff08;SEs&#xff09;在回答用户提交的查询中的作用远未被取代。例如&#xf…

idea怎么配置gradle多个版本

1.背景 gradle版本很多,而且很多时候版本是不兼容的,我们希望拉取下来的代码就包含已经配置好的版本,而不是去配置本机的gradle版本..... 意思就是要实现项目A可以用6.X版本 项目B可以使用7.X版本 项目C可以用9.X版本..... 2.配置方式 步骤一:项目根路径下保留一个文件夹…

阿里云ACP云计算高级攻城狮通用知识

&#x1f525;概述 阿里云云计算高级工程师ACP认证是面向使用阿里云云计算产品的架构、开发、运维类人员的专业技术认证&#xff0c;主要考核考生利用阿里云云计算技术服务体系设计稳定、安全、高性能、易扩展、低成本的企业云计算架构的能力。 前提&#xff1a;在写适用人群…

【ROS2】高级:从包文件读取 (C++)

目标&#xff1a;在不使用 CLI 的情况下从包中读取数据。 教程级别&#xff1a;高级 时间&#xff1a;10 分钟 目录 背景 先决条件 任务 1 创建一个包裹2 编写 C 读取器3 构建并运行 摘要 背景 rosbag2 不仅提供 ros2 bag 命令行工具。它还提供了一个 C API&#xff0c;用于从您…

基于JAVA+SpringBoot+uniapp的心理小程序(小程序版本)

✌全网粉丝20W,csdn特邀作者、博客专家、CSDN新星计划导师、java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ 技术范围&#xff1a;SpringBoot、Vue、SSM、HLMT、Jsp、SpringCloud、Layui、Echarts图表、Nodejs、爬…

5G mmWave PAAM 开发平台

Avnet-Fujikura-AMD 5G 毫米波相控阵天线模块开发平台 Avnet 和 Fujikura 为毫米波频段创建了一个领先的 5G FR2 相控阵天线开发平台。该平台使开发人员能够使用 AMD Xilinx 的 Zynq UltraScale™ RFSoC Gen3 和 Fujikura 的 FutureAcess™ 相控阵天线模块 (PAAM) 快速创建和制…

上海理工大学24计算机考研考情分析!初复试分值比55:45,复试逆袭人数不算多!

上海理工大学&#xff08;University of Shanghai for Science and Technology&#xff09;&#xff0c;位于上海市&#xff0c;是一所以工学为主&#xff0c;工学、理学、经济学、管理学、文学、法学、艺术学等多学科协调发展的应用研究型大学&#xff1b;是上海市属重点建设大…

Amisco供应汽车线圈与Husco是一家私营公司高性能液压和机电部件在汽车和非公路应用的组件设计和制造方面拥有超过 75 年的经验10于年的合作

Amisco和Husco在汽车线圈和高性能液压和机电部件的设计和制造方面合作已经超过10年。 Amisco是一家供应汽车线圈的公司&#xff0c;而Husco则专注于高性能液压和机电部件的设计和制造。 这两家公司在汽车和非公路应用领域拥有超过75年的经验。通过合作&#xff0c;Amisco和Husc…

【开源 Mac 工具推荐之 2】洛雪音乐(lx-music-desktop):免费良心的音乐平台

旧版文章&#xff1a;【macOS免费软件推荐】第6期&#xff1a;洛雪音乐 Note&#xff1a;本文在旧版文章的基础上&#xff0c;新更新展示了一些洛雪音乐的新功能&#xff0c;并且描述更为详细。 简介 洛雪音乐&#xff08;GitHub 名&#xff1a;lx-music-desktop &#xff09;…

将iPad 作为Windows电脑副屏的几种方法(二)

将iPad 作为Windows电脑副屏的几种方法&#xff08;二&#xff09; 1. 前言2. EV 扩展屏2.1 概述2.2 下载、安装、连接教程2.3 遇到的问题和解决方法2.3.1 平板连接不上电脑 3. Twomon SE3.1 概述3.2 下载安装教程 4. 多屏中心&#xff08;GlideX&#xff09;4.1 概述4.2 下载安…

LeetCode 算法:单词搜索 c++

原题链接&#x1f517;&#xff1a;单词搜索 难度&#xff1a;中等⭐️⭐️ 题目 给定一个 m x n 二维字符网格 board 和一个字符串单词 word 。如果 word 存在于网格中&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 单词必须按照字母顺序&#xff0c;通…

解决npm install(‘proxy‘ config is set properly. See: ‘npm help config‘)失败问题

摘要 重装电脑系统后&#xff0c;使用npm install初始化项目依赖失败了&#xff0c;错误提示&#xff1a;‘proxy’ config is set properly…&#xff0c;具体的错误提示如下图所示&#xff1a; 解决方案 经过报错信息查询解决办法&#xff0c;最终找到了两个比较好的方案&a…

vscode+wsl2+anaconda环境的配置与使用

目录 下载anaconda Anaconda使用参考 vscodeubuntuanaconda 先用vscode连接本地ubuntu。 如果没有安装wsl2与ubuntu&#xff0c;可点击下面的链接。 问题&#xff1a;wsl install 无法解析服务器 成功记录&#xff1a; 在vscode终端用ubuntu安装anaconda。 创建pytho…

学习008-02-01 Define the Data Model and Set the Initial Data(定义数据模型并设置初始数据)

Define the Data Model and Set the Initial Data&#xff08;定义数据模型并设置初始数据&#xff09; This section explains how to design a business model (database) for an application built with Cross-Platform .NET App UI (XAF) and Entity Framework Core. 本节…

Python简化命令行界面库之fire使用详解

概要 在开发命令行工具时,开发者通常需要编写大量代码来解析命令行参数,这既耗时又容易出错。Python Fire 是 Google 开源的一个库,旨在简化命令行界面的开发。它可以将任何 Python 对象自动生成一个命令行界面,从而大大减少了开发时间和代码复杂度。本文将详细介绍 Pytho…

HiFi-GAN——基于 GAN 的声码器,能在单 GPU 上生成 22 KHz 音频

拟议的 HiFiGAN 可从中间表征生成原始波形 源码地址&#xff1a;https://github.com/NVIDIA/DeepLearningExamples 论文地址&#xff1a;https://arxiv.org/pdf/2010.05646.pdf 研究要点包括 **挑战&#xff1a;**基于 GAN 的语音波形生成方法在质量上不及自回归模型和基于流…

排序系列 之 选择排序

&#xff01;&#xff01;&#xff01;排序仅针对于数组哦本次排序是按照升序来的哦 介绍 快速排序英文名为SelectSort从数组中找到最小的放到前边 基本思路 1、默认待排序数组中第一个作为最小值2、找待排序数组&#xff08;注意不是整个数组哦&#xff09;中真正的最小值3…

Web前端Promise

Promise介绍与使用 Promise是什么&#xff1f; 1.抽象表达&#xff1a; Promise是一门新的技术&#xff08;ES6规范&#xff09;Promise是JS中进行异步编程的新解决方案备注&#xff1a;旧方案是单纯使用回调函数 2.具体表达&#xff1a; 从语法上来说&#xff1a;Promise…

基于语音识别的会议记录系统

文章目录 核心功能页面展示使用技术方案功能结构设计数据库表展示 核心功能页面展示 视频展示功能 1.创建会议 在开始会议之前需要管理员先创建一个会议&#xff0c;为了能够快速开始会议&#xff0c;仅需填写会议的名称、会议举办小组、会议背景等简要会议信息即可成功创建。…

Apache BookKeeper 一致性协议解析

导语 Apache Pulsar 是一个多租户、高性能的服务间消息传输解决方案&#xff0c;支持多租户、低延时、读写分离、跨地域复制&#xff08;GEO replication&#xff09;、快速扩容、灵活容错等特性。Pulsar 存储层依托于 BookKeeper 组件&#xff0c;所以本文简单探讨一下 BookK…