当前位置: 首页 > 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;这篇我们来看看集成学习的另一个分支…...

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站&#xff0c;会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后&#xff0c;网站没有变化的情况。 不熟悉siteground主机的新手&#xff0c;遇到这个问题&#xff0c;就很抓狂&#xff0c;明明是哪都没操作错误&#x…...

Python爬虫实战:研究MechanicalSoup库相关技术

一、MechanicalSoup 库概述 1.1 库简介 MechanicalSoup 是一个 Python 库,专为自动化交互网站而设计。它结合了 requests 的 HTTP 请求能力和 BeautifulSoup 的 HTML 解析能力,提供了直观的 API,让我们可以像人类用户一样浏览网页、填写表单和提交请求。 1.2 主要功能特点…...

内存分配函数malloc kmalloc vmalloc

内存分配函数malloc kmalloc vmalloc malloc实现步骤: 1)请求大小调整:首先,malloc 需要调整用户请求的大小,以适应内部数据结构(例如,可能需要存储额外的元数据)。通常,这包括对齐调整,确保分配的内存地址满足特定硬件要求(如对齐到8字节或16字节边界)。 2)空闲…...

使用VSCode开发Django指南

使用VSCode开发Django指南 一、概述 Django 是一个高级 Python 框架&#xff0c;专为快速、安全和可扩展的 Web 开发而设计。Django 包含对 URL 路由、页面模板和数据处理的丰富支持。 本文将创建一个简单的 Django 应用&#xff0c;其中包含三个使用通用基本模板的页面。在此…...

VB.net复制Ntag213卡写入UID

本示例使用的发卡器&#xff1a;https://item.taobao.com/item.htm?ftt&id615391857885 一、读取旧Ntag卡的UID和数据 Private Sub Button15_Click(sender As Object, e As EventArgs) Handles Button15.Click轻松读卡技术支持:网站:Dim i, j As IntegerDim cardidhex, …...

使用分级同态加密防御梯度泄漏

抽象 联邦学习 &#xff08;FL&#xff09; 支持跨分布式客户端进行协作模型训练&#xff0c;而无需共享原始数据&#xff0c;这使其成为在互联和自动驾驶汽车 &#xff08;CAV&#xff09; 等领域保护隐私的机器学习的一种很有前途的方法。然而&#xff0c;最近的研究表明&…...

【Redis技术进阶之路】「原理分析系列开篇」分析客户端和服务端网络诵信交互实现(服务端执行命令请求的过程 - 初始化服务器)

服务端执行命令请求的过程 【专栏简介】【技术大纲】【专栏目标】【目标人群】1. Redis爱好者与社区成员2. 后端开发和系统架构师3. 计算机专业的本科生及研究生 初始化服务器1. 初始化服务器状态结构初始化RedisServer变量 2. 加载相关系统配置和用户配置参数定制化配置参数案…...

【第二十一章 SDIO接口(SDIO)】

第二十一章 SDIO接口 目录 第二十一章 SDIO接口(SDIO) 1 SDIO 主要功能 2 SDIO 总线拓扑 3 SDIO 功能描述 3.1 SDIO 适配器 3.2 SDIOAHB 接口 4 卡功能描述 4.1 卡识别模式 4.2 卡复位 4.3 操作电压范围确认 4.4 卡识别过程 4.5 写数据块 4.6 读数据块 4.7 数据流…...

linux arm系统烧录

1、打开瑞芯微程序 2、按住linux arm 的 recover按键 插入电源 3、当瑞芯微检测到有设备 4、松开recover按键 5、选择升级固件 6、点击固件选择本地刷机的linux arm 镜像 7、点击升级 &#xff08;忘了有没有这步了 估计有&#xff09; 刷机程序 和 镜像 就不提供了。要刷的时…...

相机从app启动流程

一、流程框架图 二、具体流程分析 1、得到cameralist和对应的静态信息 目录如下: 重点代码分析: 启动相机前,先要通过getCameraIdList获取camera的个数以及id,然后可以通过getCameraCharacteristics获取对应id camera的capabilities(静态信息)进行一些openCamera前的…...