当前位置: 首页 > news >正文

算法学习(十四)—— 二叉树的深度搜索(DFS)

目录

关于dfs

部分OJ题详解

2331. 计算布尔二叉树的值

129. 求根节点到叶节点数字之和

814. 二叉树剪枝

98. 验证二叉搜索树

230. 二叉搜索树中第K小的元素

257. 二叉树的所有路径

关于dfs

算法学习(十二)—— 递归,搜索,回溯前置_数组递归查找-CSDN博客

部分OJ题详解

2331. 计算布尔二叉树的值

2331. 计算布尔二叉树的值 - 力扣(LeetCode)

题目有点长,得结合示例来理解。下面来分析下这道题:

  • 看到二叉树遍历,最先想到的就是递归,这里我们用DFS来尝试解决
  • 在这道题的树中,只要节点有孩子,那必定有两个子节点,没有只有一个子节点的情况;非叶结点里面存的是“逻辑与”或“逻辑或”,叶子节点存的就是1或0,也就是“true”或“false”
  • 然后就是发现子问题的过程了,以根节点为例,假如我要知道根节点最后的值是什么,那我就得直到左子树和右子树的值是多少,然后以左子树或右子树的根节点为例,再往下进行同样的分析,就发现了我们的相同子问题,所以我们的函数头就是:bool dfs(root); 也就是你给我一个根节点,我告诉你最后的结果是true还是false
  • 然后就是函数体的设计,bool left = dfs(root->left);  bool right = dfs(root->right);  也就是要告诉我们左右子树的值,然后就是根据目前的根节点是“逻辑与”还是“逻辑或”,算出结果
  • 最后就是递归出口,就是当我遇到叶子节点的时候,直接返回true或false即可 
class Solution {
public:bool evaluateTree(TreeNode* root) {return dfs(root);    }bool dfs(TreeNode* root){if(root->left == nullptr) return root->val == 0 ? false : true;bool left = dfs(root->left);bool right = dfs(root->right);if(root->val == 2) return left | right;else return left & right;}
};

129. 求根节点到叶节点数字之和

 129. 求根节点到叶节点数字之和 - 力扣(LeetCode)

下面来分析下这道题:

  • 我们仍然还是尝试用递归的dfs来尝试解决下,还是那三个步骤:找相同子问题,搞清楚每个子问题要做什么,递归出口
  • 首先是相同子问题,上面示例的树太小了,我们换个大的树,如下图:
  • 所以函数头设计为:int dfs(root, presum); 也就是传一个根节点,以及这个根节点前面路径的值presum传给后面,然后给我返回这个根节点最后计算的值
  • 函数体的步骤就是:①先算出该根节点加上前面路径的计算结果值    ②把这个值传给左子树    ③把这个值传给右子树    ④左右子树返回计算结果 
  • 递归出口就是当遇到叶子节点时,把上面的第一步计算完之后,再正式返回,不再将这个值传给左右子树
class Solution {
public:int sumNumbers(TreeNode* root) {return dfs(root, 0);}int dfs(TreeNode* root, int prevsum){prevsum = prevsum * 10 + root->val; //①if(root->left == nullptr && root->right == nullptr) return prevsum; //递归出口int ret = 0;if(root->left != nullptr) ret += dfs(root->left, prevsum); //②if(root->right != nullptr) ret += dfs(root->right, prevsum); //③return ret; //④}
};

814. 二叉树剪枝

814. 二叉树剪枝 - 力扣(LeetCode)

题目的意思很明确,就是如果一个节点的子树中没有值为1的节点的话,就把这个子树干掉,下面我们来分析下这道题:

  • 这个题目和我们之前的题目有点不一样,前面的题目就是让我们根据信息求值,没有改变二叉树,而这个题目是要求我们实实在在的对二叉树做了修改了。还是用递归的dfs来尝试解决这道题还是三个问题
  • 先来看下如何找到相同的子问题:当我遍历到一个节点时,我们要做的就是“这个节点是否应该被剪掉”,依据就是“这个节点的子树是否含有值为1的节点”,所以这道题一定是后序遍历,因为只有后序遍历才有机会回到根节点的时候带着左右子树的信息
  • 当遍历到叶节点时,已经没有左右子树了,并且如果我自己也是0,就把我自己干掉,回到父节点后,还要修改下父节点的指针,这一步可以通过返回值解决,就是把父节点的指针修改成这个返回值即可,所以函数头需要有一个返回值Node*
  • 如果已经是叶节点,但我自己是1,那么就不需要把我干掉,就把我自己的指针向上返回,就一直通过上面的步骤,就可以干掉所有符合要求的子树,所以函数头的设计就是:Node* dfs(root);
  • 然后就是函数体的设置,基本步骤和上面的一样:①处理左子树    ②处理右子树    ③判断是否为叶结点并且判断是否为1,进行剪枝操作并返回
  • 最后是递归出口,就是当root == null 的时候,直接返回
class Solution {
public:TreeNode* pruneTree(TreeNode* root) {return dfs(root);}TreeNode* dfs(TreeNode* root){if(root == nullptr) return root;root->left = dfs(root->left);root->right = dfs(root->right);if(root->left == nullptr && root->right == nullptr && root->val == 0)root = nullptr;return root;}
};

98. 验证二叉搜索树

98. 验证二叉搜索树 - 力扣(LeetCode)

 

这道题我们主要是理解三个点:①理解全局变量的优势    ②回溯    ③剪枝;后面会给出两种思路和代码,一个是有剪枝的,一种没有剪枝,下面来分析下这道题:

  •  二叉搜索树的中序遍历结果,是一个有序的序列,所以,我们可以验证这棵树的中序遍历结果是否有序即可,这个可以用一个全局变量来搞
  • 先搞一个全局变量 “prev = 负无穷大”,然后每次中序遍历时,都更新一下 prev的值,以上面的示例2为例,根据中序遍历,假设遍历到4的位置,prev此时的值为3,然后将此时节点的值和prev作比较验证有序即可,最后就是遍历到5,prev再次更新为4,再次比较即可,所以只要和prev作比较然后返回true或false即可

策略一,也就是上面的方法:

  • 验证左子树是二叉搜索树
  • 当前遍历的节点符合二叉搜索树的定义
  • 验证右子树是二叉搜索树 

策略二,剪枝:

  •  还是以示例二为例,假设递归已经完成了验证左子树是二叉搜索树,此时prev是5
  • 当中序遍历到3时,3小于5,所以已经不符合二叉搜索树的定义了,此时我们直接一路返回false到根节点,不再去比较节点6了,这就是剪枝,意义就是加快搜索过程

无剪枝代码:

class Solution 
{
public:long prev = LONG_MIN;bool isValidBST(TreeNode* root) {return dfs(root);  }bool dfs(TreeNode* root){if(root == nullptr) return true;bool left = dfs(root->left);bool cur = false; //cur表示当前节点是否满足二叉搜索树条件if(root->val > prev) cur = true;prev = root->val;bool right = dfs(root->right);return left && right && cur;}
};

 剪枝:

class Solution 
{
public:long prev = LONG_MIN;bool isValidBST(TreeNode* root) {return dfs(root);  }bool dfs(TreeNode* root){if(root == nullptr) return true;bool left = dfs(root->left);if(left == false) return false; //剪枝:就是当左子树已经不符合搜索二叉树了,直接返回bool cur = false; //cur代表当前节点是否满足二叉搜索树条件if(root->val > prev) cur = true;if(cur == false) return false; //如果我当前位置节点已经不符合二叉搜索树要求了,直接返回,不再去遍历右子树了,这就是剪枝prev = root->val;bool right = dfs(root->right);return left && right && cur;}
};

230. 二叉搜索树中第K小的元素

230. 二叉搜索树中第 K 小的元素 - 力扣(LeetCode)

这道题题意还是很简单的,也是给我们一个二叉搜索树,然后让我们返回这个树的节点中值第K小的值,下面来分析下这道题:

  • 有了上一道题的经验之后,这道题其实超级简单,因为我们已经知道了“二叉搜索树的中序遍历结果是有序的”,所以我们就可以用“ 两个全局遍历 + 中序遍历 ”就可以解决这道题
  • 以上面的示例二为例,搞两个全局遍历,count = k = 3,ret = 0,ret的作用就是当count一直减减到0时,用ret标记一下当前节点的值然后返回
  • 这里可以用剪枝,就是当count为0时,直接返回后面的步骤都不再执行,这就是剪枝,这一步加个判断即可

 

class Solution {
public:int count;int ret;int kthSmallest(TreeNode* root, int k) {count = k;dfs(root);return ret;}void dfs(TreeNode* root){if(root == nullptr || count <= 0) return;dfs(root->left);count--;if(count == 0) ret = root->val;dfs(root->right);}
};

257. 二叉树的所有路径

257. 二叉树的所有路径 - 力扣(LeetCode)

前面两道题我们提到了三个词:①全局变量    ②回溯    ③剪枝,其中回溯有个性质叫做“恢复现场”

关于恢复现场:

网上很多人在自学算法的时候,都是看到了代码中有“恢复现场”的操作,才知道这道题用了“回溯”,所以很多人以为的就是“恢复现场 --> 回溯

而今天,我们就要把这个因果关系给反过来:

  • 如果有递归,那么肯定有回溯:递归 --> 回溯
  • 如果有回溯,那么肯定有“恢复现场”:回溯 --> 恢复现场
  • 所以,如果有递归,那么就有恢复现场:递归 --> 恢复现场

在后面的难度较高的递归回溯题目的时候,有了“回溯 --> 恢复现场”的思路之后,写代码的过程才会有思路,并且,一旦我们在题目中用到了全局变量,那么恢复现场的操作就会显得很重要

这个题目很简单,就是让我们把每一个路径保存在一个字符串数组里面,具体步骤通过示例一和示例二很好理解,下面来分析下这道题:

  • 思路很简单,就是在搜索的过程中,记录每一个节点即可,当遇到叶子节点的时候停止搜索
  • 我们可以搞两个全局变量,①一个字符串数组 string[] ret:就是在我遍历时发现叶子节点时,就把结果扔 ret 里去;    ②一个字符串变量 string path:作用就是在深度遍历的时候,记录一下路径,以示例一为例,遍历到1时,path的值为“1->”;当遍历到2时,path的值为“1->2>”;当遍历到5时,由于5是叶节点,所以只添加数字,所以最后path的值为“1->2->5”
  • 但是前面说过,有递归就有回溯,当遍历完5,回溯到1时,接下来就是要往右边回溯,但是此时path的值不是从1开始的“1->”,而是还是之前的“1->2->5”,所以我们需要“恢复现场”,就是我们在向上返回的时候,需要不断pop掉path的结尾,但是又因为是全局path,需要很多很多次的添加和删除,不太好控制,所以这道题我们不用全局的path,后面再用
  • 那么用什么代替全局path呢?我们可以在函数头设置一下:void dfs(root, path);  如果我们把path当作参数传递,那么就是每次调用path的时候,相当于重新创建了一个path变量,也就是每次深度搜索时,都用的是自己单独地path变量,代替了“恢复现场”的操作和功能
  • 然后激素关心一下每一个子问题在干什么,也激素函数体的设置:如果root是叶子节点,只在path后面添加上数字即可;如果root不是叶子节点,dfs(root->left, path); dfs(root->right, path)
  • 最后就是递归出口,就是注意一下剪枝:当左子树为空时,就不需要再 dfs(root->left)了,直接去搞右子树即可:if(root == nullptr) return;  
class Solution {
public:vector<string> ret;vector<string> binaryTreePaths(TreeNode* root) {string path;if(root == nullptr) return ret;dfs(root, path);return ret;}void dfs(TreeNode* root, string path){path += to_string(root->val);if(root->left == nullptr && root->right == nullptr) //如果是叶子节点,直接返回,不再搜索{ret.push_back(path);return;}path += "->";if(root->left) dfs(root->left, path);if(root->right) dfs(root->right, path);}
};

相关文章:

算法学习(十四)—— 二叉树的深度搜索(DFS)

目录 关于dfs 部分OJ题详解 2331. 计算布尔二叉树的值 129. 求根节点到叶节点数字之和 814. 二叉树剪枝 98. 验证二叉搜索树 230. 二叉搜索树中第K小的元素 257. 二叉树的所有路径 关于dfs 算法学习&#xff08;十二&#xff09;—— 递归&#xff0c;搜索&#xff0c…...

【vue2】封装自定义的日历组件(三)之基础添加月份的加减定位到最新月份的第一天

我们在切换月份的时候&#xff0c;希望高亮显示在每个月的第一天上面&#xff0c;这样的效果我们要怎么来实现&#xff0c;其实也很简单&#xff0c;我们先看下实现的效果 实现效果 代码实现 原理就是获取到每月的第一天日期&#xff0c;然后再跟整个的数据进行对比&#xff…...

LabVIEW偏心圆筒流变仪测控系统

偏心圆筒流变仪是一种专门研究聚合物熔体在复杂流场中特殊流变行为的先进设备。通过结合硬件控制与LabVIEW软件开发&#xff0c;本系统实现了对流变仪功能的精准控制与数据采集&#xff0c;进一步提高了聚合物加工过程的研究精度和效率。 项目背景 传统的流变测量设备多集中于…...

Runloop

假设你的项目中有关tableView&#xff0c;然后还有一个定时器timer在执行&#xff0c;定时器代码如下&#xff1a; var num 0override func viewDidLoad() {super.viewDidLoad()let timer Timer(timeInterval: 1,target: self,selector: #selector(self.run),userInfo: nil,r…...

SpringBoot的Bean类三种注入方式(附带LomBok注入)

SpringBoot的Bean类三种注入方式&#xff08;附带LomBok注入&#xff09; 在 Spring Boot 中&#xff0c;Bean 的注入方式主要包括构造函数注入&#xff08;Constructor Injection&#xff09;、字段注入&#xff08;Field Injection&#xff09;以及 Setter 方法注入&#xf…...

开源向量数据库介绍说明

开源向量数据库 Milvus 特点&#xff1a;分布式、高性能&#xff0c;支持亿级向量检索。 支持的数据类型&#xff1a;文本、图像、音频、视频等。 使用场景&#xff1a;推荐系统、语义搜索、图像搜索。 数据存储后端&#xff1a;支持多种后端&#xff0c;如 SQLite、MySQL、Pos…...

【前端】深度解析 JavaScript 中的 new 关键字与构造函数

博客主页&#xff1a; [小ᶻ☡꙳ᵃⁱᵍᶜ꙳] 本文专栏: 前端 文章目录 &#x1f4af;前言&#x1f4af;构造函数的核心特性&#x1f4af;new 关键字的执行机制&#x1f4af;实例代码与详细解析代码示例代码逐步解析 &#x1f4af;new 的内部执行模拟执行过程的详细解析 &am…...

2024年华中杯数学建模C题基于光纤传感器的平面曲线重建算法建模解题全过程文档及程序

2024年华中杯数学建模 C题 基于光纤传感器的平面曲线重建算法建模 原题再现 光纤传感技术是伴随着光纤及光通信技术发展起来的一种新型传感器技术。它是以光波为传感信号、光纤为传输载体来感知外界环境中的信号&#xff0c;其基本原理是当外界环境参数发生变化时&#xff0c…...

使用 `typing_extensions.TypeAlias` 简化类型定义:初学者指南

使用 typing_extensions.TypeAlias 简化类型定义&#xff1a;初学者指南 什么是 TypeAlias&#xff1f;安装 typing_extensions示例代码&#xff1a;如何使用 TypeAlias示例 1&#xff1a;为简单类型定义别名示例 2&#xff1a;为复杂类型定义别名示例 3&#xff1a;结合 Union…...

如何快速批量把 PDF 转为 JPG 或其它常见图像格式?

在某些特定场景下&#xff0c;将 PDF 转换为 JPG 图片格式却具有不可忽视的优势。例如&#xff0c;当我们需要在不支持 PDF 查看的设备或软件中展示文档内容时&#xff0c;JPG 图片能够轻松被识别和打开&#xff1b;此外&#xff0c;对于一些网络分享或社交媒体发布的需求&…...

如何在组织中塑造和强化绩效文化?

在组织中塑造和强化绩效文化是一个系统性的工程。 一、明确绩效目标与期望 设定清晰目标 组织应根据自身战略规划&#xff0c;将长期目标分解为具体、可衡量、可实现、相关联、有时限&#xff08;SMART&#xff09;的短期和中期绩效目标。例如&#xff0c;一家连锁餐饮企业的…...

OllyDbg、CE简单介绍

基础知识&#xff1a; 想要破解软件&#xff0c;需要一些基础知识&#xff1a; 文件格式&#xff1a;Windows对应PE、Linux对应ELF、IOS对应Mash-0。文件格式是指操作系统规定的每个段&#xff08;代码段、数据段、堆、栈&#xff09;的大小、顺序等信息。 汇编语言&#xff1…...

Python函数——函数的返回值定义语法

一、引言 在Python中&#xff0c;函数的返回值是其核心功能之一&#xff0c;它使得函数能够将计算结果传递给调用者&#xff0c;进而推动程序的逻辑和功能实现。理解和掌握函数的返回值语法&#xff0c;不仅能够提高代码的模块化和可读性&#xff0c;还能使程序更加高效和灵活…...

【Pandas】pandas isna

Pandas2.2 General Top-level missing data 方法描述isna(obj)用于检测数据中的缺失值isnull(obj)用于检测数据中的缺失值notna(obj)用于检测数据中的非缺失值notnull(obj)用于检测数据中的非缺失值 pandas.isna() pandas.isna() 是 Pandas 库中的一个函数&#xff0c;用于…...

mysql 数据库表的大小

mysql 数据库表的大小 Mysql 查看数据库各个表占用空间 mysql如何查看数据库所有表大小 在MySQL中&#xff0c;要查看数据库所有表的大小&#xff0c;可以使用以下方法&#xff1a; 方法一&#xff1a;使用information_schema数据库 首先&#xff0c;通过命令行或图形界面…...

(6)JS-Clipper2之ClipperOffset

1. 描述 ClipperOffset类封装了对打开路径和关闭路径进行偏移(膨胀/收缩)的过程。 这个类取代了现在已弃用的OffsetPaths函数&#xff0c;该函数不太灵活。可以使用不同的偏移量(增量)多次调用Execute方法&#xff0c;而不必重新分配路径。现在可以在一次操作中对开放和封闭路…...

如何在Ubuntu中利用repo和git地址下载获取imx6ull的BSP

01-设置git的用户名和邮箱 git config --global user.name "suwenhao" git config --global user.email "2487872782qq.com"这里不设置的话后面在第5步的repo配置中还是会要求输入&#xff0c;而且以后进行相关操作都要输入&#xff0c;不妨现在就进行配置…...

Ruby On Rails 笔记5——常用验证下

3.Validation Options 3.1 :allow_nil 当验证值为nil时:allow_nil选项会跳过验证 class Coffee < ApplicationRecordvalidates :size, inclusion: { in: %w(small medium large),message: "%{value} is not a valid size" }, allow_nil: true end irb> Cof…...

JS听到了因果的回响

这是我学习JS的第11天了&#xff0c;&#xff0c;&#xff0c;我现在赶着周末学JS&#xff0c;然后还有二十多天就期末了呵呵呵。。。 图片切换模块 思路分析&#xff1a; 这是实现的代码&#xff0c;建议还是把不同的变量定义出来比较合适&#xff1a; //获取三个盒子// 小盒…...

【高中生讲机器学习】28. 集成学习之 Bagging 随机森林!

创建时间&#xff1a;2024-12-09 首发时间&#xff1a;2024-12-09 最后编辑时间&#xff1a;2024-12-09 作者&#xff1a;Geeker_LStar 嘿嘿&#xff0c;你好呀&#xff01;我又来啦~~ 前面我们讲完了集成学习之 Boooooosting&#xff0c;这篇我们来看看集成学习的另一个分支…...

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析

1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具&#xff0c;该工具基于TUN接口实现其功能&#xff0c;利用反向TCP/TLS连接建立一条隐蔽的通信信道&#xff0c;支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式&#xff0c;适应复杂网…...

Chapter03-Authentication vulnerabilities

文章目录 1. 身份验证简介1.1 What is authentication1.2 difference between authentication and authorization1.3 身份验证机制失效的原因1.4 身份验证机制失效的影响 2. 基于登录功能的漏洞2.1 密码爆破2.2 用户名枚举2.3 有缺陷的暴力破解防护2.3.1 如果用户登录尝试失败次…...

SkyWalking 10.2.0 SWCK 配置过程

SkyWalking 10.2.0 & SWCK 配置过程 skywalking oap-server & ui 使用Docker安装在K8S集群以外&#xff0c;K8S集群中的微服务使用initContainer按命名空间将skywalking-java-agent注入到业务容器中。 SWCK有整套的解决方案&#xff0c;全安装在K8S群集中。 具体可参…...

golang循环变量捕获问题​​

在 Go 语言中&#xff0c;当在循环中启动协程&#xff08;goroutine&#xff09;时&#xff0c;如果在协程闭包中直接引用循环变量&#xff0c;可能会遇到一个常见的陷阱 - ​​循环变量捕获问题​​。让我详细解释一下&#xff1a; 问题背景 看这个代码片段&#xff1a; fo…...

前端倒计时误差!

提示:记录工作中遇到的需求及解决办法 文章目录 前言一、误差从何而来?二、五大解决方案1. 动态校准法(基础版)2. Web Worker 计时3. 服务器时间同步4. Performance API 高精度计时5. 页面可见性API优化三、生产环境最佳实践四、终极解决方案架构前言 前几天听说公司某个项…...

论文解读:交大港大上海AI Lab开源论文 | 宇树机器人多姿态起立控制强化学习框架(一)

宇树机器人多姿态起立控制强化学习框架论文解析 论文解读&#xff1a;交大&港大&上海AI Lab开源论文 | 宇树机器人多姿态起立控制强化学习框架&#xff08;一&#xff09; 论文解读&#xff1a;交大&港大&上海AI Lab开源论文 | 宇树机器人多姿态起立控制强化…...

VM虚拟机网络配置(ubuntu24桥接模式):配置静态IP

编辑-虚拟网络编辑器-更改设置 选择桥接模式&#xff0c;然后找到相应的网卡&#xff08;可以查看自己本机的网络连接&#xff09; windows连接的网络点击查看属性 编辑虚拟机设置更改网络配置&#xff0c;选择刚才配置的桥接模式 静态ip设置&#xff1a; 我用的ubuntu24桌…...

Git 3天2K星标:Datawhale 的 Happy-LLM 项目介绍(附教程)

引言 在人工智能飞速发展的今天&#xff0c;大语言模型&#xff08;Large Language Models, LLMs&#xff09;已成为技术领域的焦点。从智能写作到代码生成&#xff0c;LLM 的应用场景不断扩展&#xff0c;深刻改变了我们的工作和生活方式。然而&#xff0c;理解这些模型的内部…...

LLaMA-Factory 微调 Qwen2-VL 进行人脸情感识别(二)

在上一篇文章中,我们详细介绍了如何使用LLaMA-Factory框架对Qwen2-VL大模型进行微调,以实现人脸情感识别的功能。本篇文章将聚焦于微调完成后,如何调用这个模型进行人脸情感识别的具体代码实现,包括详细的步骤和注释。 模型调用步骤 环境准备:确保安装了必要的Python库。…...

零基础在实践中学习网络安全-皮卡丘靶场(第十一期-目录遍历模块)

经过前面几期的内容我们学习了很多网络安全的知识&#xff0c;而这期内容就涉及到了前面的第六期-RCE模块&#xff0c;第七期-File inclusion模块&#xff0c;第八期-Unsafe Filedownload模块。 什么是"遍历"呢&#xff1a;对学过一些开发语言的朋友来说应该知道&…...