0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

二叉树的所有路径介绍

新材料在线 来源:代码随想录 作者:程序员Carl 2021-08-13 17:51 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

以为只用了递归,其实还用了回溯

257. 二叉树的所有路径

题目地址:https://leetcode-cn.com/problems/binary-tree-paths/

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

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

思路

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

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

前序遍历以及回溯的过程如图:

07b5afe6-fbbe-11eb-9bcf-12bb97331649.png

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

递归

递归函数函数参数以及返回值

要传入根节点,记录每一条路径的path,和存放结果集的result,这里递归不需要返回值,代码如下:

void traversal(TreeNode* cur, vector《int》& path, vector《string》& result)

确定递归终止条件

再写递归的时候都习惯了这么写:

if (cur == NULL) {

终止处理逻辑

}

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

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

所以本题的终止条件是:

if (cur-》left == NULL && cur-》right == NULL) {

终止处理逻辑

}

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

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

这里使用vector结构path来记录路径,所以要把vector结构的path转为string格式,在把这个string 放进 result里。

那么为什么使用了vector结构来记录路径呢? 因为在下面处理单层递归逻辑的时候,要做回溯,使用vector方便来做回溯。

可能有的同学问了,我看有些人的代码也没有回溯啊。

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

这里我们先使用vector结构的path容器来记录路径,那么终止处理逻辑如下:

if (cur-》left == NULL && cur-》right == NULL) { // 遇到叶子节点

string sPath;

for (int i = 0; i 《 path.size() - 1; i++) { // 将path里记录的路径转为string格式

sPath += to_string(path[i]);

sPath += “-》”;

}

sPath += to_string(path[path.size() - 1]); // 记录最后一个节点(叶子节点)

result.push_back(sPath); // 收集一个路径

return;

}

确定单层递归逻辑

因为是前序遍历,需要先处理中间节点,中间节点就是我们要记录路径上的节点,先放进path中。

path.push_back(cur-》val);

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

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

if (cur-》left) {

traversal(cur-》left, path, result);

}

if (cur-》right) {

traversal(cur-》right, path, result);

}

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

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

if (cur-》left) {

traversal(cur-》left, path, result);

}

if (cur-》right) {

traversal(cur-》right, path, result);

}

path.pop_back();

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

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

那么代码应该这么写:

if (cur-》left) {

traversal(cur-》left, path, result);

path.pop_back(); // 回溯

}

if (cur-》right) {

traversal(cur-》right, path, result);

path.pop_back(); // 回溯

}

那么本题整体代码如下:

class Solution {private:

void traversal(TreeNode* cur, vector《int》& path, vector《string》& result) {

path.push_back(cur-》val);

// 这才到了叶子节点

if (cur-》left == NULL && cur-》right == NULL) {

string sPath;

for (int i = 0; i 《 path.size() - 1; i++) {

sPath += to_string(path[i]);

sPath += “-》”;

}

sPath += to_string(path[path.size() - 1]);

result.push_back(sPath);

return;

}

if (cur-》left) {

traversal(cur-》left, path, result);

path.pop_back(); // 回溯

}

if (cur-》right) {

traversal(cur-》right, path, result);

path.pop_back(); // 回溯

}

}

public:

vector《string》 binaryTreePaths(TreeNode* root) {

vector《string》 result;

vector《int》 path;

if (root == NULL) return result;

traversal(root, path, result);

return result;

}

};

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

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

class Solution {private:

void traversal(TreeNode* cur, string path, vector《string》& result) {

path += to_string(cur-》val); // 中

if (cur-》left == NULL && cur-》right == NULL) {

result.push_back(path);

return;

}

if (cur-》left) traversal(cur-》left, path + “-》”, result); // 左

if (cur-》right) traversal(cur-》right, path + “-》”, result); // 右

}

public:

vector《string》 binaryTreePaths(TreeNode* root) {

vector《string》 result;

string path;

if (root == NULL) return result;

traversal(root, path, result);

return result;

}

};

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

注意在函数定义的时候void traversal(TreeNode* cur, string path, vector《string》& result) ,定义的是string path,每次都是复制赋值,不用使用引用,否则就无法做到回溯的效果。

那么在如上代码中,貌似没有看到回溯的逻辑,其实不然,回溯就隐藏在traversal(cur-》left, path + “-》”, result);中的 path + “-》”。 每次函数调用完,path依然是没有加上“-》” 的,这就是回溯了。

为了把这份精简代码的回溯过程展现出来,大家可以试一试把:

if (cur-》left) traversal(cur-》left, path + “-》”, result); // 左 回溯就隐藏在这里

改成如下代码:

path += “-》”;

traversal(cur-》left, path, result); // 左

即:

if (cur-》left) {

path += “-》”;

traversal(cur-》left, path, result); // 左

}

if (cur-》right) {

path += “-》”;

traversal(cur-》right, path, result); // 右

}

此时就没有回溯了,这个代码就是通过不了的了。

如果想把回溯加上,就要 在上面代码的基础上,加上回溯,就可以AC了。

if (cur-》left) {

path += “-》”;

traversal(cur-》left, path, result); // 左

path.pop_back(); // 回溯

path.pop_back();

}

if (cur-》right) {

path += “-》”;

traversal(cur-》right, path, result); // 右

path.pop_back(); // 回溯

path.pop_back();

}

大家应该可以感受出来,如果把 path + “-》”作为函数参数就是可以的,因为并有没有改变path的数值,执行完递归函数之后,path依然是之前的数值(相当于回溯了)

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

迭代法

至于非递归的方式,我们可以依然可以使用前序遍历的迭代方式来模拟遍历路径的过程,对该迭代方式不了解的同学,可以看文章二叉树:听说递归能做的,栈也能做!和二叉树:前中后序迭代方式统一写法。

这里除了模拟递归需要一个栈,同时还需要一个栈来存放对应的遍历路径。

C++代码如下:

class Solution {public:

vector《string》 binaryTreePaths(TreeNode* root) {

stack《TreeNode*》 treeSt;// 保存树的遍历节点

stack《string》 pathSt; // 保存遍历路径的节点

vector《string》 result; // 保存最终路径集合

if (root == NULL) return result;

treeSt.push(root);

pathSt.push(to_string(root-》val));

while (!treeSt.empty()) {

TreeNode* node = treeSt.top(); treeSt.pop(); // 取出节点 中

string path = pathSt.top();pathSt.pop(); // 取出该节点对应的路径

if (node-》left == NULL && node-》right == NULL) { // 遇到叶子节点

result.push_back(path);

}

if (node-》right) { // 右

treeSt.push(node-》right);

pathSt.push(path + “-》” + to_string(node-》right-》val));

}

if (node-》left) { // 左

treeSt.push(node-》left);

pathSt.push(path + “-》” + to_string(node-》left-》val));

}

}

return result;

}

};

当然,使用java的同学,可以直接定义一个成员变量为object的栈Stack《Object》 stack = new Stack《》();,这样就不用定义两个栈了,都放到一个栈里就可以了。

总结

本文我们开始初步涉及到了回溯,很多同学过了这道题目,可能都不知道自己其实使用了回溯,回溯和递归都是相伴相生的。

我在第一版递归代码中,把递归与回溯的细节都充分的展现了出来,大家可以自己感受一下。

第二版递归代码对于初学者其实非常不友好,代码看上去简单,但是隐藏细节于无形。

最后我依然给出了迭代法。

对于本地充分了解递归与回溯的过程之后,有精力的同学可以在去实现迭代法。

其他语言版本

Java:

//解法一class Solution {

/**

* 递归法

*/

public List《String》 binaryTreePaths(TreeNode root) {

List《String》 res = new ArrayList《》();

if (root == null) {

return res;

}

List《Integer》 paths = new ArrayList《》();

traversal(root, paths, res);

return res;

}

private void traversal(TreeNode root, List《Integer》 paths, List《String》 res) {

paths.add(root.val);

// 叶子结点

if (root.left == null && root.right == null) {

// 输出

StringBuilder sb = new StringBuilder();

for (int i = 0; i 《 paths.size() - 1; i++) {

sb.append(paths.get(i)).append(“-》”);

}

sb.append(paths.get(paths.size() - 1));

res.add(sb.toString());

return;

}

if (root.left != null) {

traversal(root.left, paths, res);

paths.remove(paths.size() - 1);// 回溯

}

if (root.right != null) {

traversal(root.right, paths, res);

paths.remove(paths.size() - 1);// 回溯

}

}

}

Python

class Solution:

def binaryTreePaths(self, root: TreeNode) -》 List[str]:

path=[]

res=[]

def backtrace(root, path):

if not root:return

path.append(root.val)

if (not root.left)and (not root.right):

res.append(path[:])

ways=[]

if root.left:ways.append(root.left)

if root.right:ways.append(root.right)

for way in ways:

backtrace(way,path)

path.pop()

backtrace(root,path)

return [“-》”.join(list(map(str,i))) for i in res]

Go:

func binaryTreePaths(root *TreeNode) []string {

res := make([]string, 0)

var travel func(node *TreeNode, s string)

travel = func(node *TreeNode, s string) {

if node.Left == nil && node.Right == nil {

v := s + strconv.Itoa(node.Val)

res = append(res, v)

return

}

s = s + strconv.Itoa(node.Val) + “-》”

if node.Left != nil {

travel(node.Left, s)

}

if node.Right != nil {

travel(node.Right, s)

}

}

travel(root, “”)

return res

}

责任编辑:haq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 函数
    +关注

    关注

    3

    文章

    4424

    浏览量

    68155
  • 二叉树
    +关注

    关注

    0

    文章

    74

    浏览量

    13039

原文标题:二叉树的所有路径:不止递归,还有回溯

文章出处:【微信号:xincailiaozaixian,微信公众号:新材料在线】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    UniStore正式全面开放

    5月7日,宇科技正式宣布,全球首个人形机器人任务动作应用商店——宇UniStore官方共享应用平台即日起面向全球用户全面开放。用户可通过该平台开发和下载机器人应用,像安装手机App一样简单,无需任何底层编程能力。
    的头像 发表于 05-09 11:22 1139次阅读

    高速数字电路回流路径怎么找

    回流路径沿着阻抗最低的路径流动,高频时就是信号线下方的参考平面。
    的头像 发表于 04-11 17:11 1299次阅读
    高速数字电路回流<b class='flag-5'>路径</b>怎么找

    科技,IPO申请获受理

    电子发烧友网综合报道 3月20日,上交所网站显示,宇科技股份有限公司科创板IPO申请已受理,成为又一家科创板IPO“预先审阅”落地项目。此次IPO,宇科技拟募资42.02亿元。   招股书显示
    的头像 发表于 03-20 18:33 4448次阅读
    宇<b class='flag-5'>树</b>科技,IPO申请获受理

    输入引脚时钟约束_Xilinx FPGA编程技巧-常用时序约束详解

    基本的约束方法 为了保证成功的设计,所有路径的时序要求必须能够让执行工具获取。最普遍的三种路径以及异常路径为: 输入路径(Input Path),使用输入约束 寄存器到寄存器
    发表于 01-16 08:19

    TÜV莱茵与杭集团达成战略合作并颁发欧盟CE-MD符合性证书

    日前,国际独立第三方检测、检验和认证机构德国莱茵TÜV大中华区(简称"TÜV莱茵")与杭集团股份有限公司(简称"杭集团")签署了战略合作协议,标志着双方
    的头像 发表于 01-15 12:18 547次阅读

    亿纬锂能与杭集团达成战略合作

    近日,亿纬锂能与杭集团2025年战略研讨会暨战略合作协议签约仪式在杭州举行。亿纬锂能副总裁、商用车电池产品线总裁江吉兵博士,亿纬锂能商用车电池产品线国内销售部总经理井振江,杭集团董事、副总经理兼
    的头像 发表于 01-04 18:18 1239次阅读

    Termux中调试圣诞Python代码

    : python --version 如果输出Python 3.x.x(比如3.11.4),说明安装成功。 、代码编写(两种方式可选) 方式1:用Termux自带编辑器(nano)(新手推荐) 创建并编辑
    发表于 12-09 09:02

    通过优化代码来提高MCU运行效率

    选择时间复杂度低的算法。 根据访问模式选择数据结构。频繁查找用哈希表,有序数据用二叉树等。 查表法:对于复杂的数学计算(如sin, log),或者协议解析,预先计算好结果存于数组中,用空间换时间
    发表于 11-12 08:21

    Verilog实现使用Booth编码和Wallace的定点补码乘法器原理

    的“和”位继续在本列传播,这就构成了Wallace Tree乘法器。 Wallace充分利用全加器3-2压缩的特性,随时将可利用的所有输入和中间结果及时并行计算,大大节省了计算延时。下图
    发表于 10-23 08:01

    蜂鸟E203内核中断管理模块sirv_plic_man代码分析

    。 上面的代码生成一个二叉树结构来比较和选择具有最大优先级的挂起中断源及其ID。树状结构由级联比较器组成,每一层的比较器数量是前一层的一半。在的每一层,选择优先级最高的中断并传递到下一层,直到只剩下
    发表于 10-23 06:05

    请问rtt studio 的文件夹打红什么意思?

    rtt studio 的文件夹打红什么意思?而且文件夹里面实际是有文件的,但是浏览不出来。
    发表于 09-18 06:34

    科技,被起诉

    电子发烧友网综合报道 天眼查显示,近日,杭州宇科技股份有限公司(以下简称“宇科技”)新增1条开庭公告,原告为杭州露韦美日化有限公司(以下简称“露韦美日化”),案由为侵害发明专利权纠纷,该案将于8
    的头像 发表于 08-26 07:50 5340次阅读
    宇<b class='flag-5'>树</b>科技,被起诉

    亿纬锂能荣获杭集团2022-2024年度优秀供应商奖

    近日,亿纬锂能凭借卓越产品、可靠交付与优质服务荣获杭集团颁发的“2022-2024年度优秀供应商”奖。杭集团副总经理兼杭电器董事长金华曙、杭电器总经理兼杭博电机总经理李明辉出席
    的头像 发表于 07-15 09:00 1171次阅读

    想在rtsmart中使用uart2,是不是只能通过修改设备方法来实现uart2的复用呀?

    我想在rtsmart中使用uart2,是不是只能通过修改设备方法来实现uart2的复用呀? 修改设备后如何只编译设备文件? 编译生成的文件可以直接替换到庐山派里吗,具体替换路径
    发表于 06-24 07:04

    电源工程师的核心技能体系

    电源工程师的核心技能体系需覆盖从基础理论到专业实践、工具应用及行业适配的全链条能力。以下是系统化的技能框架,按知识层级和应用场景展开,帮助从业者明确能力提升路径: 一、基础理论层:核心知识根基
    的头像 发表于 06-05 09:44 3654次阅读