LeetCode 257. 二叉树的所有路径(回溯详解)

文章目录

  • LeetCode 257. 二叉树的所有路径
    • 思路
    • 递归
      • 版本一:非常明确的回溯代码
      • 版本二:精简的回溯代码

LeetCode 257. 二叉树的所有路径

LeetCode 257. 二叉树的所有路径

给定一个二叉树,返回所有从根节点到叶子节点的路径。

说明: 叶子节点是指没有子节点的节点。

示例:

输入:

   1/   \
2     3\5

输出: ["1->2->5", "1->3"]

解释: 所有根节点到叶子节点的路径为: 1->2->5, 1->3

思路

这道题目要求从根节点到叶子的路径,所以需要前序遍历,这样才方便让父节点指向孩子节点,找到对应的路径。

在这道题目中将第一次涉及到回溯,因为我们要把路径记录下来,需要回溯来回退一个路径再进入另一个路径。

前序遍历以及回溯的过程如图:
在这里插入图片描述

回溯和递归中回归的区别:
回溯:因为我们要把路径记录下来,需要回溯来回退一个路径再进入另一个路径。
回归:下层函数栈执行完成后,回归到上层函数栈来

实际上,回溯和回归都是伴随递归出现的,只是回溯比回归多了一个路径回退的操作。

我们先使用递归的方式,来做前序遍历。要知道递归和回溯就是一家的,本题也需要回溯。

递归

  1. 递归函数参数以及返回值
    要传入根节点root,记录每一条路径的str,和存放结果集的res,这里递归不需要返回值,代码如下:
func dfs(root *TreeNode,res *[]string,str string) {}
  1. 确定递归终止条件
    在写递归的时候都习惯了这么写:
if root == nil {终止处理逻辑
}

但是本题的终止条件这样写会很麻烦,因为本题要找到叶子节点,就开始结束的处理逻辑了(把路径放进res里)。

那么什么时候算是找到了叶子节点? 是当 root不为空,其左右孩子都为空的时候,就找到叶子节点。

所以本题的终止条件是:

if root.Left == nil && root.Right == nil {终止处理逻辑}

为什么没有判断root是否为空呢,因为下面的逻辑可以控制空节点不入循环。

再来看一下终止处理的逻辑。

这里使用string类型str来记录路径,注意在下面处理单层递归逻辑的时候,要做回溯,可能有的同学问了,我看有些人的代码也没有回溯啊。

其实是有回溯的,只不过隐藏在函数调用时的参数赋值里(值传递),下文我还会提到。

这里我们使用string类型的str来记录路径,那么终止处理逻辑如下:

if root.Left == nil && root.Right == nil {*res = append(*res,str)}
  1. 确定单层递归逻辑
    因为是前序遍历,需要先处理中间节点,中间节点就是我们要记录路径上的节点,先加到str中。
str = str + "->" + fmt.Sprintf("%d",root.Left.Val)

然后是递归和回溯的过程,上面说过没有判断root是否为空,那么在这里递归的时候,如果为空就不进行下一层递归了。

所以递归前要加上判断语句,下面要递归的节点是否为空,如下

 if root.Left != nil {dfs(root.Left,res,str)}if root.Right != nil {dfs(root.Right,res,str)}

此时还没完,递归完,要做回溯啊,因为str 不能只一直加入节点,它还要删节点,然后才能加入新的节点。

那么回溯要怎么回溯呢,一些同学会这么写,如下:

if root.Left != nil {dfs(root.Left,res,str)}if root.Right != nil {dfs(root.Right,res,str)}l := len("->" + fmt.Sprintf("%d",root.Left.Val))str = str[0:len(str)-l] // 回溯

这个回溯就有很大的问题,我们知道,回溯和递归是一一对应的,有一个递归,就要有一个回溯,这么写的话相当于把递归和回溯拆开了, 一个在花括号里,一个在花括号外。

所以回溯要和递归永远在一起,世界上最遥远的距离是你在花括号里,而我在花括号外!

那么代码应该这么写:

if root.Left != nil {str = str + "->" + fmt.Sprintf("%d",root.Left.Val)dfs(root.Left,res,str)l := len("->" + fmt.Sprintf("%d",root.Left.Val))str = str[0:len(str)-l] // 回溯}if root.Right != nil {str = str + "->" + fmt.Sprintf("%d",root.Right.Val)dfs(root.Right,res,str)l := len("->" + fmt.Sprintf("%d",root.Right.Val))str = str[0:len(str)-l] // 回溯}

那么本题整体Go代码如下:

版本一:非常明确的回溯代码

/*** Definition for a binary tree node.* type TreeNode struct {*     Val int*     Left *TreeNode*     Right *TreeNode* }*/
func binaryTreePaths(root *TreeNode) []string {if root == nil {return []string{}}res := make([]string,0)str := fmt.Sprintf("%d",root.Val)dfs(root,&res,str)return res
}func dfs(root *TreeNode,res *[]string,str string) {if root.Left == nil && root.Right == nil {*res = append(*res,str)}if root.Left != nil {str = str + "->" + fmt.Sprintf("%d",root.Left.Val)dfs(root.Left,res,str)l := len("->" + fmt.Sprintf("%d",root.Left.Val))str = str[0:len(str)-l] // 回溯}if root.Right != nil {str = str + "->" + fmt.Sprintf("%d",root.Right.Val)dfs(root.Right,res,str)l := len("->" + fmt.Sprintf("%d",root.Right.Val))str = str[0:len(str)-l] // 回溯}
}

在这里插入图片描述

如上的Go代码充分体现了回溯。

那么如上代码可以精简成如下代码:

版本二:精简的回溯代码

/*** Definition for a binary tree node.* type TreeNode struct {*     Val int*     Left *TreeNode*     Right *TreeNode* }*/
func binaryTreePaths(root *TreeNode) []string {if root == nil {return []string{}}res := make([]string,0)str := fmt.Sprintf("%d",root.Val)dfs(root,&res,str)return res
}func dfs(root *TreeNode,res *[]string,str string) {if root.Left == nil && root.Right == nil {*res = append(*res,str)}if root.Left != nil {dfs(root.Left,res,str + "->" + fmt.Sprintf("%d",root.Left.Val))}if root.Right != nil {dfs(root.Right,res,str + "->" + fmt.Sprintf("%d",root.Right.Val))}
}

在这里插入图片描述

如上代码精简了不少,也隐藏了不少东西。

注意在函数定义的时候func dfs(root *TreeNode,res *[]string,str string) ,定义的是str string ,每次都是复制赋值,不用使用指针,否则就无法做到回溯的效果。

那么在如上代码中,貌似没有看到回溯的逻辑,其实不然,回溯就隐藏在 dfs(root.Left,res,str + "->" + fmt.Sprintf("%d",root.Left.Val))
中的str + "->" + fmt.Sprintf("%d",root.Left.Val)。 每次函数调用完,str依然是没有加上"->" + fmt.Sprintf("%d",root.Left.Val) 的,这就是回溯了。

综合以上,第二种递归的代码虽然精简但把很多重要的点隐藏在了代码细节里,第一种递归写法虽然代码多一些,但是把每一个逻辑处理都完整的展现出来了。

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

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

相关文章

全网最适合入门的面向对象编程教程:51 Python函数方法与接口-使用Zope实现接口

全网最适合入门的面向对象编程教程:51 Python 函数方法与接口-使用 Zope 实现接口 摘要: 在 Python 中,Zope 提供了一种机制来定义和实现接口。Zope 的接口模块通常用于创建可重用的组件,并确保组件遵循特定的接口规范。 原文链…

力扣 209.长度最小的子数组

一、长度最小的子数组 二、解题思路 采用滑动窗口的思路&#xff0c;详细见代码。 三、代码 class Solution {public int minSubArrayLen(int target, int[] nums) {int n nums.length, left 0, right 0, sum 0;int ans n 1; for (right 0; right < n; right ) { …

【二等奖论文】2024年华为杯研赛D题成品论文(后续会更新)

您的点赞收藏是我继续更新的最大动力&#xff01; 一定要点击如下的卡片&#xff0c;那是获取资料的入口&#xff01; 点击链接获取【2024华为杯研赛资料汇总】&#xff1a; https://qm.qq.com/q/jTIeGzwkSchttps://qm.qq.com/q/jTIeGzwkSc 题 目&#xff1a; 大数据驱动的…

一劳永逸:用脚本实现夸克网盘内容自动更新

系统环境&#xff1a;debian/ubuntu 、 安装了python3 原作者项目&#xff1a;https://github.com/Cp0204/quark-auto-save 感谢 缘起 我喜欢看电影追剧&#xff0c;会经常转存一些资源到夸克网盘&#xff0c;电影还好&#xff0c;如果是电视剧&#xff0c;麻烦就来了。 对于一…

深度学习-卷积神经网络(CNN)

文章目录 一、网络构造1. 卷积层&#xff08;Convolutional Layer&#xff09;&#xff08;1&#xff09;卷积&#xff08;2&#xff09;特征图计算公式&#xff08;3&#xff09;三通道卷积 2. 激活函数&#xff08;Activation Function&#xff09;3. 池化层&#xff08;Pool…

【JUC并发编程系列】深入理解Java并发机制:线程局部变量的奥秘与最佳实践(五、ThreadLocal原理、对象之间的引用)

文章目录 【JUC并发编程系列】深入理解Java并发机制&#xff1a;线程局部变量的奥秘与最佳实践(五、ThreadLocal原理、对象之间的引用)1. 基本 API 介绍2. 简单用法3. 应用场景4. Threadlocal与Synchronized区别5. 内存溢出和内存泄漏5.2 内存溢出 (Memory Overflow)5.2 内存泄…

全栈项目小组【算法赛】题目及解题

题目&#xff1a;全栈项目小组【算法赛】 题目&#xff1a; 解题思路 1.遍历简历信息&#xff1a;我们需要读取所有简历&#xff0c;根据期望薪资和岗位类型进行分类和统计。 2.分类统计&#xff1a;使用哈希表来存储每个薪资下的前端&#xff08;F&#xff09;和后端&#…

【线程】线程的同步

本文重点&#xff1a;理解条件变量和生产者消费者模型 同步是在保证数据安全的情况下&#xff0c;让我们的线程访问资源具有一定的顺序性 条件变量cond 当一个线程互斥地访问某个变量时&#xff0c;它可能发现在其它线程改变状态之前&#xff0c;它什么也做不了&#xff0c;…

window系统DockerDesktop 部署windows容器

目录 参考文献1、安装Docker Desktop1.1 下载安装包1.2 安装教程1.3 异常解决 2、安装windows容器2.1 先启动DockerDesktop 软件界面2.2 检查docker版本2.3 拉取windows镜像 参考文献 windows容器docker中文官网 Docker: windows下跑windows镜像 1、安装Docker Desktop 1.1 …

SSM框架VUE电影售票管理系统开发mysql数据库redis设计java编程计算机网页源码maven项目

一、源码特点 smm VUE电影售票管理系统是一套完善的完整信息管理类型系统&#xff0c;结合SSM框架和VUE、redis完成本系统&#xff0c;对理解vue java编程开发语言有帮助系统采用ssm框架&#xff08;MVC模式开发&#xff09;&#xff0c;系 统具有完整的源代码和数据库&#…

【C语言零基础入门篇 - 17】:排序算法

文章目录 排序算法排序的基本概念冒泡排序选择排序插入排序 排序算法 排序的基本概念 1、什么是排序&#xff1f; 排序是指把一组数据以某种关系&#xff08;递增或递减&#xff09;按顺序排列起来的一种算法。 例如&#xff1a;数列 8、3、5、6、2、9、1、0、4、7 递增排序…

深入浅出:Eclipse 中配置 Maven 与 Spark 应用开发全指南

Spark 安装配置 1.在 Eclipse 中配置 Maven Eclipse 中默认自带 Maven 插件&#xff0c;但是自带的 Maven 插件不能修改本地仓库&#xff0c;所 以通常我们不使用自带的 Maven &#xff0c;而是使用自己安装的&#xff0c;在 Eclipse 中配置 Maven 的 步骤如下&#xff1a;…

Nature Electronics |无感佩戴的纤维基电子皮肤(柔性半导体器件/柔性健康监测/电子皮肤/柔性传感/纤维器件)

英国剑桥大学Yan Yan Shery Huang课题组,在《Nature Electronics 》上发布了一篇题为“Imperceptible augmentation of living systems with organic bioelectronic fibres”的论文,第一作者为王文宇博士(Wenyu Wang),论文内容如下: 一、 摘要 利用电子技术对人类皮肤和…

0-PCIE串行高速接口架构介绍

随着计算机技术日新月异的发展&#xff0c;对于I/O传输速率的需求愈发提高&#xff0c;PCI总线由于是并行传输&#xff0c;在时钟频率提高之后会带来信号偏移和串扰的问题从而使信号衰减失真&#xff0c;同时在数据传输速率不断提高之后PCI总线还面临着管脚限制&#xff0c;传输…

哈电集团数智化转型新突破:浪潮信息SAP HANA驱动数智升级

浪潮信息SAP HANA一体化解决方案&#xff0c;鼎力推动哈尔滨电气集团有限公司&#xff08;哈电集团&#xff09;取得了数字化转型的非凡成就。该定制化方案不仅促使哈电集团业财一体化程度显著跃升&#xff0c;突破70%大关&#xff0c;更确保了库存管理的绝对精准&#xff0c;库…

【C++前缀和 排序】2171. 拿出最少数目的魔法豆|1748

本文涉及的基础知识点 C算法&#xff1a;前缀和、前缀乘积、前缀异或的原理、源码及测试用例 包括课程视频 LeetCode2171. 拿出最少数目的魔法豆 难度分&#xff1a;1748 给定一个 正整数 数组 beans &#xff0c;其中每个整数表示一个袋子里装的魔法豆的数目。 请你从每个袋…

Vue3实现类ChatGPT聊天式流式输出(vue-sse实现)

1. 效果展示 流式输出 直接输出 2. 核心代码 找了一些示例与AI生成的代码&#xff0c;或多或少有些问题&#xff0c;搞了好久&#xff0c;郁闷~&#xff0c;在此记录下 2.1 依赖安装 npm install vue-sse2.2 改写main.ts import VueSSE from vue-sseconst app Vue.cre…

饲料颗粒机全套设备有哪些机器组成

饲料颗粒机全套设备通常包括原料粉碎、混合机、制粒机、冷却器、筛分机、包装机以及配套的电气控制等多个部分组成&#xff1a;1、粉碎机&#xff1a;将各种饲料原料进行清理、去杂、破碎等预处理&#xff0c;确保原料的纯净度和适宜粒度&#xff0c;为后续加工做准备。2、混合…

撤销与恢复的奥秘:设计模式之备忘录模式详解

备忘录模式 &#x1f3af; 备忘录模式&#xff08;Memento Pattern&#xff09;简介 备忘录模式 是一种行为型设计模式&#xff0c;用于保存对象的某一时刻状态&#xff0c;以便稍后可以恢复到该状态&#xff0c;而不破坏对象的封装性。备忘录模式将对象的状态封装在一个独立的…

240922-Conda的在线下载与离线安装

A. 修改路径&#xff08;如果需要&#xff09; 在 conda 中无法直接通过命令指定下载路径。默认情况下&#xff0c;conda 将软件包下载到其缓存目录中&#xff0c;具体位置通常是 ~/miniconda/pkgs 或 ~/anaconda/pkgs&#xff0c;取决于你安装 conda 的路径。 如果你希望将下…