颠沛流离学二叉树(完结撒花篇)

本篇会加入个人的所谓鱼式疯言

❤️❤️❤️鱼式疯言:❤️❤️❤️此疯言非彼疯言

而是理解过并总结出来通俗易懂的大白话,

小编会尽可能的在每个概念后插入鱼式疯言,帮助大家理解的.

🤭🤭🤭可能说的不是那么严谨.但小编初心是能让更多人能接受我们这个概念 !!!

在这里插入图片描述

前言

在上两篇 二叉树系列 的文章中,我们学习了二叉树的 概念特性遍历方式 , 而在本篇文章中,将展开对 二叉树其他基本核心操作 的详解 ❤️ ❤️ ❤️ ❤️

提前透露一下哦, 本篇文章的基本操作,小编都会带着大家用 递归非递归 的代码带着大家走一遍哦 💖 💖 💖 💖

目录

  1. 获取树中节点的个数

  2. 获取叶子节点的个数

  3. 获取第 K 层的节点的个数

  4. 获取二叉树的高度(最大深度)

  5. 查找 节点的值等于的 value 的节点

  6. 判断一棵树是否是完全二叉树

一. 获取树中节点的个数

1. 操作分析

由于我们的二叉树是分为 两边的,分别是 左子树右子树 , 要求树中的节点,我们就需要知道 左子树的节点右子树的节点 ,最后加上自身的 1个根节点

在这里插入图片描述

2. 代码实现

<1>. 递归实现

// 递归计算节点个数
public int Size(TreeNode root) {if (root==null) {return 0;}return Size(root.left)+1+Size(root.right);
}

递归的代码实现,就是咱们 动画上的执行过程 了,小编在这里就不赘述了 💞 💞 💞 💞

动画展示

请添加图片描述

在这里插入图片描述

<2>. 非递归实现

// 非递归遍历节点个数public int nodeSize;public void setNodeSize(TreeNode root) {if (root==null) {return;}// 只要遍历过的节点// 就让 节点数 ++nodeSize++;// 先遍历左子树setNodeSize(root.left);// 遍历右子树setNodeSize(root.right);}

在这里插入图片描述

非递归的执行过程 :

我们这里利用了 前序遍历 的思路, 当我们每走过一个节点,就让我们的 nodeSize++

当完全的走过 每一个节点 , 我们 的 nodeSize 就加多少次

当我们遍历完 整个二叉树 , 也就统计出二叉树 所有的节点

鱼式疯言

关于求树中接节点个数的操作,小编最大的体会就是 子问题递归思路

由于我们 递归 的是不断把 每一个节点重新看成是根节点 的思路

所以我们只需要求出每个根节点的个数。所累积的个数,不就是我们的 节点总数 吗 ! ! !

这就是 子问题 的思路, 小伙伴在思考这一类问题时,一定要学会把一个 大问题 拆分成我们能解决的 一个一个小问题 的思想.

二. 获取叶子节点的个数

1. 操作分析

小伙伴都知道我们 二叉树的叶子节点 的定义,就是 度为0 的节点

那么我们该怎么找 度为零 的节点呢 💞 💞 💞 💞

其实很简单,我们只需要返回 左右子树所有节点 同时 左节点右子树 也为 null

最后 左右子树的叶子节点相加 即可

在这里插入图片描述

2. 代码实现

<1>. 递归实现

// 递归计算叶子个数
public int leafSize(TreeNode root) {if (root==null) {return 0;}if (root.left==null && root.right==null) {return 1;}return leafSize(root.left)+leafSize(root.right);
}

在这里插入图片描述

关于 递归实现 的执行步骤,下面就让小编用 动画 的形式来描述吧 😊 😊 😊 😊

动画展示

请添加图片描述

<2>. 非递归实现

public int leafSize;// 非递归求叶子节点的个数public void setLeafSize(TreeNode root) {if (root==null) {return;}// 如果左右节点为 null// 就是我们需要的节点if (root.right==null && root.left==null) {leafSize++;}// 遍历左子树setLeafSize(root.left);// 遍历右子树setLeafSize(root.right);}

在这里插入图片描述

非递归的实现过程

非递归的根本是依赖于 我们的遍历 来实现的,当遍历到某个 左节点为 null 并且 右节点也为 null 时, 就令 leafSize ++

当遍历完整个 二叉树的所有节点,也就意味着 所有的叶子节点 已经被 leafSize 统计完了

鱼式疯言

对于 递归实现 的思路来说,我们还是用 子问题 的思路,把一棵树拆成 多颗子树

子树 又成为 新的树, 在新的树里面去寻找 左右子树都为null 的节点 ,

最终回归到 最开始的整棵树 上解决该问题 ! ! !

三. 获取第 K 层的节点的个数

1. 操作分析

要得到 第 K 层的节点 , 不免想起我们上一篇文章中所用的 层序遍历 的思想, 每一层也就相当于 二维数组中的每一行

所有要得到 每一层的节点数 ,本质上就是当我们 递归往下递的每走一行的 左子树和右子树 , k 就 减去一行 。

最终 当 K = 1 时, 就回退到 整棵树的根节点 ,就是我们要 获取到 第 K 层的节点个数

在这里插入图片描述

2. 代码实现

 // 求树某深度的节点数public int treeIndexlength(TreeNode root , int key) {// 如果节点为 null 返回 0if (root==null) {return 0;}/**** 当 k = 1 时 就意味着到底最后一层* 返回当前根节点即可****/// 如果 这个位置 没有左右if (key==1) {return 1;}/*** 先统计左边的节点** 再统计右边的节点** 最后都要减少 key 次* 代表统计了一层** 也就是说每统计一层 ,就减少一层**///int left = treeIndexlength(root.left,key-1);// 再统计右边的节点int right = treeIndexlength(root.right,key-1);// 一直返回该 key 位置的节点return left+right;}

在这里插入图片描述

具体 递归的过程 我们通过下面动画展示:请添加图片描述

鱼式疯言

扩展一下

小伙伴有没有想到一个点:

   // 一直返回该 key 位置的节点return left+right;

我们这边是 直接返回 left + right 是得到我们当前 第k层 深度的节点数

但如果我们

 return left+right + 1;  

我们如果是 + 1 呢?

答案就是返回 从整棵树的根节点第k层所有节点总和

感兴趣的小伙伴可以自己 研究 哦, 小编在这里就不细讲了 💖 💖 💖 💖**(代码如下)**

// 求树某深度的到根节点所有节点数
public int treeIndexSize(TreeNode root , int key) {if (root==null) {return 0;}if (key==1) {return 1;}int left=treeIndexSize(root.left,key-1);int right=treeIndexSize(root.right,key-1);// key 位置开始查找节点 并 一直累加return left+right+1;
}

四.获取树的高度(最大深度)

1. 操作分析

要获取高度小伙伴们先得理解高度该怎么来, 我们可以设想一下,

如果一颗树的左子树高度为 3 右子树的高度为 4 , 那么 这颗树的高度是 3 还是 4

很显然 这都 不对

树的高度最终应该是 5 才对, 因为啊, 我们树的高度是 左右子树中较高的那个高度 再 + 上自身根节点的高度 1 , 所以我们得到的树的高度是 4 + 1 才对

在这里插入图片描述

2. 代码实现

// 返回树的最大深度
public int treeHeight(TreeNode root) {if (root==null) {return 0;}// 求左子树的深度int leftnum=treeHeight(root.left);// 求右子树的深度int rightnum=treeHeight(root.right);/*** 根据左右子树两边的高度** 哪边高 取 哪边*  最后还要加上该节点的高度**/return leftnum>rightnum? leftnum+1: rightnum+1;
}

在这里插入图片描述

具体我们由于递归不好描述,请小伙伴们看动画分析递归的全过程哦 💫 💫 💫 💫

动画展示

请添加图片描述

鱼式疯言

当我们用 子问题 的思路 在 寻找数的高度 的关键在于我们要注意加上不仅要比较 两端左子树右子树高度, 那个自身的 根节点的 1 也要同时加上

五. 查找节点的值等于的 value 的节点

1. 操作分析

查找某个等于 val 的值, 我们就需要在 遍历的基础上 完成

所以我们只需要考虑 一点 即可,当我们在遍历过程

如果 左子树或者右子树 的节点的 val 找到了等于 val 就返回该 节点 , 如果没找到就返回 null .

在这里插入图片描述

2. 代码实现

// 查找节点public TreeNode find(TreeNode root, char val) {if(root==null) {return null;}// 判断新的根节点的 val 是否 等于目标值 valif (Character.compare(val,root.val)==0) {return  root;}// 在左子树中查找该节点TreeNode node1=  find(root.left,val);// 一旦找到就返回该节点if (node1 !=null) {return node1;}// 在右子树中查找该今节点TreeNode node2= find(root.right,val);// 一点找到就返回该节点if (node2 !=null) {return node2;}// 都没有找到就返回 nullreturn  null;}

在这里插入图片描述

下面我们看看动画对于这段代码的理解吧 🤔 🤔 🤔 🤔

动画展示

请添加图片描述

鱼式疯言

细节处理:

   // 在左子树中查找该节点TreeNode node1=  find(root.left,val);// 一旦找到就返回该节点if (node1 !=null) {return node1;}// 在右子树中查找该今节点TreeNode node2= find(root.right,val);// 一点找到就返回该节点if (node2 !=null) {return node2;}

我想肯定有小伙伴会直接返回 node1 或者 node2 的值, 而不加判断, 这样做是不是有点问题呢 ???

是的,是有些问题,小伙伴应该知道如果我们先 递归左子树 或者 右子树 来达到遍历的效果, 一但 左子树没有找到该 val目标值 ,就会去右边的 右子树

p但我们 中途一旦返回,就会陷入只找寻 左边一个节点 ,然后就直接返回了,未达到遍历的效果 , 也就更不可能找到我们的 val 值了

六. 判断一棵树是否是完全二叉树

1. 操作分析

要考虑是否是完全二叉树,我们只需要考虑一点即可:

左子树 < 右子树高度 : 用 -1 来标记,代表 不是完全二叉树

左子树 >= 右子树高度: 就返回该高度

在这里插入图片描述

2. 代码实现

/*** 检查左子树和右子树的高度关系* * 当左子树 > 右子树时 ,就不是完全二叉树* 否则就是完全二叉树* @param root 根节点* @return 返回布尔类型* */private int CheckTreeCount(TreeNode root) {if (root==null) {return  0;}int left=CheckTreeCount(root.left);/*** 当此树不是完全二叉树的就返回 负数* * 而这里的判断是为了连接上一个递归下来的根节点* */// 当左子树不是完全二叉树时,就返回 -1 if (left < 0) {return -1;}int right=CheckTreeCount(root.right);// 当右子树不是完全二叉树时, 就返回 -1 if(right < 0) {return  -1;}// 当左子树的高度 < 右子树的高度// 就说明不是完全二叉树if (left < right) {return -1;} else {// 该节点是完全二叉树时,就返回当前高度return Math.max(left,right)+1;}}

在这里插入图片描述

具体实现,请小伙伴们观看下面动态理解哦 ❣️ ❣️ ❣️ ❣️

请添加图片描述

鱼式疯言

int left=CheckTreeCount(root.left);/*** 当此树不是完全二叉树的就返回 负数* * 而这里的判断是为了连接上一个递归下来的根节点* */// 当左子树不是完全二叉树时,就返回 -1 if (left < 0) {return -1;}int right=CheckTreeCount(root.right);// 当右子树不是完全二叉树时, 就返回 -1 if(right < 0) {return  -1;}

细节处理

这里有小伙伴就问了,我可以把

   if (left < 0) {return -1;}

if(right < 0) {return  -1;}

去掉吗?

答案是 不行

因为当我们一旦判断出 有一颗子树 的 左子树的高度 > 右子树的高度 时 , 就必须返回 -1 ,然后就必须回退到上一个 根节点

如果没有及时返回 -1 ,就有可能造成 max(-1,-1) 的情况,这种情况是 不可预料 的,所以我们呢需要 及时返回

总结

  1. 获取树中节点的个数: 我们通过都用了 子问题遍历 的获取节点的个数,但子问题要 注意的是需要加上自身的那个 节点数的 1

  2. 获取叶子节点的个数 : 获取叶子节点,我们主要是根据它自身的特点 (左右节点是否同时为 null ) ,成立时我们就返回 当前节点 .

  3. 获取第 K 层的节点的个数 : 获取 第K层的节点数 的思路在遍历的情况下,每走一层,k就减去1 , 当 k= 1 时就到达该层,返回 该层节点数 即可

  4. 获取二叉树的高度(最大深度):求树的高度,本质是抓住我们左子树和右子树的 较高 的那个高度,然后 加上 自身的高度1 即可

  5. 查找 节点的值等于的 value 的节点 :查找 val 是在遍历的情况下完成的,只需要在 边遍历边查找一旦找到就返回该节点即可

  6. 判断一棵树是否是完全二叉树 : 完全二叉树的判断是建立在获取二叉树高度的思路上进行变通的, 根据完全二叉树成立条件: 左子树 >= 右子树 ,这个特点来判断即可

如果觉得小编写的还不错的咱可支持 三连 下 (定有回访哦) , 不妥当的咱请评论区 指正

希望我的文章能给各位宝子们带来哪怕一点点的收获就是 小编创作 的最大 动力 💖 💖 💖

在这里插入图片描述

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

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

相关文章

文献解读-肿瘤测序-第五期|《局部晚期或转移性儿童及青少年分化型甲状腺癌的基因特征与临床特征及131I疗效的关系》

关键词&#xff1a;应用遗传流行病学&#xff1b;群体测序&#xff1b;肿瘤测序&#xff1b; 文献简介 标题&#xff08;英文&#xff09;&#xff1a;The relationship between genetic characteristics and clinical characteristics and the efficacy of 131I therapy in c…

2024年端午节放假通知

致尊敬的客户以及全体同仁&#xff1a; 2024年端午节将至&#xff0c;根据国务院办公厅通知精神&#xff0c;结合公司的实际情况&#xff0c;现将放假事宜通知如下&#xff1a; 2024年6月8日&#xff08;星期六&#xff09;至6月10日&#xff08;星期一&#xff09;&#xff…

vue3 setup 使用 beforeRouteEnter 组件内路由守卫

vue3 setup 使用 beforeRouteEnter 组件内路由守卫 setup 中只有onBeforeRouteLeave、onBeforeRouteUpdate两个钩子函数&#xff0c; 没有beforeRouteEnter对应的钩子函数&#xff0c;所以无法在setup中直接使用 <script setup> onBeforeRouteLeave((to, from) > {// …

PMP考试难吗?考试通过率有多少?

我们通常以考试的通过率来评判一个考试的难易程度。通常通过率达到60%以上&#xff0c;这个考试就不太难&#xff1b;达到80% &#xff0c;这个考试就是不难的。 PMP考试难吗&#xff1f; 不少想要考PMP的小伙伴都会有这样的疑惑&#xff0c;首先以PMP的含金量来说&#xff0…

人工智能--深度神经网络

目录 &#x1f349;引言 &#x1f349;深度神经网络的基本概念 &#x1f348;神经网络的起源 &#x1f34d; 神经网络的基本结构 &#x1f349;深度神经网络的结构 &#x1f348; 卷积神经网络&#xff08;CNN&#xff09; &#x1f348;循环神经网络&#xff08;RNN&…

车来了冲刺上市:业绩波动明显,依赖广告业务,滴滴、阿里入股

近日&#xff0c;MetaLight Inc.&#xff08;下称“元光科技”或“车来了”&#xff09;向港交所递交招股说明书&#xff0c;中金公司为其独家保荐人。 据招股书介绍&#xff0c;元光科技专注于利用时序数据&#xff08;按时间顺序排列的数据点&#xff09;来发现及预测分析对…

css网格背景样式

空白内容效果图 在百度页面测试效果 ER图效果 注意&#xff1a;要给div一个宽高 <template><div class"grid-bg"></div> </template><style scoped> .grid-bg {width: 100%;height: 100%;background: url(data:image/svgxml;base…

Tomcat概述及部署

目录 一、Tomcat概述 1.Tomcat的简介 2.Tomcat 核心的三个组件 3.应用场景 4.Tomcat 请求过程 二、部署安装Tomcat 三、Tomcat 虚拟主机配置 四、Tomcat多实例部署 一、Tomcat概述 1.Tomcat的简介 Tomcat 是 Java 语言开发的&#xff0c;Tomcat 服务器是一个免费的开…

Vuforia AR篇(六)— Mid Air 半空识别

目录 前言一、什么是Mid Air&#xff1f;二、使用步骤三、示例代码四、效果 前言 增强现实&#xff08;AR&#xff09;技术正在改变我们与数字世界的互动方式。Vuforia作为先进的AR开发平台&#xff0c;提供了多种工具来创造引人入胜的AR体验。其中&#xff0c;Mid Air功能以其…

EMC整改学习-笔记

EMC整改学习-笔记 来自赛盛技术的笔记 如果我拿到一个产品超标的一个频谱图的话&#xff0c;首先我们可以对比做一个分析。来确定你干扰源的一个分类和定义是哪些。是你这个产品类型&#xff0c;什么样的电路对应什么样的一个。从我们的一个大量的一个测试数据的经验来看&…

yolov5的口罩识别系统+GUI界面 (附代码)

基于YOLOv5模型的口罩识别系统&#xff0c;结合了GUI界面&#xff0c;旨在帮助用户快速、准确地识别图像或视频中佩戴口罩的情况。YOLOv5是一种流行的目标检测模型&#xff0c;具有高效的实时检测能力&#xff0c;而GUI界面则提供了友好的用户交互界面&#xff0c;使得整个系统…

用易查分制作活动抽奖系统,支持随机分配中奖结果!

学校或企业在开展抽奖活动时&#xff0c;如何确保公平公正&#xff0c;随机挑选中奖人员呢&#xff1f; 易查分的预置数据分配功能就可以实现&#xff0c;并且支持提交信息后随机分配中奖结果&#xff0c;不受任何人为因素的影响。下面就来教大家如何制作吧。 &#x1f4cc;使用…

Vxe UI vxe-upload vue上传组件,显示进度条的方法

vxe-upload vue 上传组件 查看官网 https://vxeui.com 显示进度条很简单&#xff0c;需要后台支持进度就可以了&#xff0c;后台实现逻辑具体可以百度&#xff0c;这里只介绍前端逻辑。 vue 上传附件 相关参数说明&#xff0c;具体可以看文档&#xff1a; multiple 是否允许…

ChatTTS+Python编程搞定语音报时小程序

文字转语音神器Python编程搞定语音报时小程序 今天一个好哥们发了一个文字转语音的AI神器的短视频。这个神器的网站是[ChatTTS - Text-to-Speech for Conversational Scenarios][https://chattts.com/]&#xff0c;如下图所示&#xff1a; 这个开源项目可以从github.com上下载…

【python】成功解决“NameError: name ‘X’ is not defined”错误的全面指南

成功解决“NameError: name ‘X’ is not defined”错误的全面指南 一、引言 在Python编程中&#xff0c;NameError: name X is not defined是一个常见的错误。这个错误通常意味着我们试图使用一个未定义的变量名X。本文将详细解析这一错误的原因&#xff0c;并提供一系列实用…

【Qt】Frame和Widget的区别

1. 这两个伙计有啥区别&#xff1f; 2. 区别 2.1 Frame继承自Widget&#xff0c;多了一些专有的功能 Frame Widget 2.2 Frame可以设置边框

【YOLO系列】YOLOv10论文超详细解读(翻译 +学习笔记)

前言 研究AI的同学们面对的一个普遍痛点是&#xff0c;刚开始深入研究一项新技术&#xff0c;没等明白透彻&#xff0c;就又迎来了新的更新版本——就像我还在忙着逐行分析2月份发布的YOLOv9代码&#xff0c;5月底清华的大佬们就推出了全新的v10。。。 在繁忙之余&#xff0…

视频监控业务平台LntonCVS国标GB28181视频平台智慧城市应用方案

随着科技的不断进步&#xff0c;尤其是人工智能技术的飞速发展&#xff0c;视频应用已经超越了传统的视频监控、视频会议、视频通话和视频指挥调度等基本功能。它们正在向更加多元化、灵活化、融合化和智能化的方向发展。因此&#xff0c;建立一个视频AI中台变得至关重要。 通过…

智慧冶金:TSINGSEE青犀AI+视频技术助力打造高效、安全的生产环境

一、建设背景 冶金行业因其特殊的生产环境和工艺要求&#xff0c;对安全生产、环境保护以及质量监控等方面有着极高的要求。因此&#xff0c;将视频智能监控技术引入冶金行业&#xff0c;不仅有助于提升生产效率&#xff0c;更能有效保障生产安全&#xff0c;降低事故风险。 …

密码学基本概念(补充)

BiBa模型的*特性规则&#xff1a;主体不能修改更高完整级的客体&#xff08;主题不能向上写&#xff09; Diffie-Hellman密钥交换协议的安全性基于求解离散对数的困难性&#xff0c;既对于C^d M mod P&#xff0c;在已知C和P的前提下&#xff0c;由d求M很容易&#xff0c;但是…