news 2026/6/15 17:31:36

二叉树--所有路径

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
二叉树--所有路径

很明显本题是一个回溯算法,解题的时候要注意,递归的下一句就是回溯。同时本题提供了两种不同的解法。

显式回溯 vs 隐式回溯

特性隐式回溯显式回溯写法 (vector<int> 或 string&)
参数传递string path(值传递,拷贝)vector<int>& path(引用传递)
内存开销较大。每层递归都要拷贝整个字符串。较小。全程共用一个容器。
回溯操作。利用函数作用域自动销毁副本。必须手动pop_back()
代码示例traversal(node, path + "->", res)path.push_back(val); traversal(...); path.pop_back();
理解难度简单,符合直觉。需要理解“恢复现场”的概念。

显式回溯

显式回溯将回溯的过程写了出来,比较容易理解。

tips:

  1. path使用整型数组是因为方便回溯。
  2. path和result是引用类,意味这整个递归全程共用一个容器。
  3. 使用先序遍历访问二叉树,是因为先序遍历符合二叉树的访问顺序。
  4. 根节点处理在递归结束条件前,是因为叶子节点也要加入path。
  5. 回溯就在递归的后面。

代码

void backtracking_1(TreeNode* root, vector<int>& path, vector<string>& result) { path.push_back(root->val); // 1. 处理当前节点 // 2. 判断叶子节点 if (root->left == nullptr && root->right == nullptr) { string sPath; for (int i = 0; i < path.size() - 1; i++) { sPath += to_string(path[i]); sPath += "->"; } sPath += to_string(path.back()); result.push_back(sPath); return; // 遇到叶子节点,记录路径后返回(注意:此时不pop,由调用层处理) } // 3. 递归逻辑 if (root->left) { backtracking_1(root->left, path, result); path.pop_back(); // 回溯:处理完左子树,把左孩子弹出,恢复现场 } if (root->right) { backtracking_1(root->right, path, result); path.pop_back(); // 回溯:处理完右子树,把右孩子弹出 } }

显式回溯整体比较好理解。

隐式回溯

隐式回溯的核心在于利用 C++ 的“值传递” (Pass by Value) 特性。

  1. 引用传递 (&):全程共享同一个 string 对象,子递归的修改会影响父递归,因此需要手动 pop_back 来“显式回溯”。
  2. 值传递 (无 &):每次递归都会拷贝一份新的 string 副本。子递归只修改自己的副本,不影响父递归。
  3. 当子递归函数返回时,副本自动销毁,父递归的变量保持原样,从而实现了“隐式回溯”。

注意代码中的注释,解释为了为什么可以实现隐式回溯。

代码

void backtracking_2(TreeNode* root, string path, vector<string>& result){ // 这里修改的是本层的path值,也不会影响上一层的path值,但是会影响传递给下一层的path值 path += to_string(root->val); if(!root->left && !root->right){ result.push_back(path); return; } if(root->left){ // path + "->":传递给下一层的path值为本层的path值加上"->",但是本层的path没有改变 backtracking_2(root->left, path + "->", result); } if(root->right){ backtracking_2(root->right, path + "->", result); } }

拓展

当你理解了隐式回溯,下面的代码你应该也可以理解。

class Solution { private: void traversal(TreeNode* cur, string path, vector<string>& result) { path += to_string(cur->val); // 中,中为什么写在这里,因为最后一个节点也要加入到path中 if (cur->left == NULL && cur->right == NULL) { result.push_back(path); return; } 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(); // 回溯 '-' } } public: vector<string> binaryTreePaths(TreeNode* root) { vector<string> result; string path; if (root == NULL) return result; traversal(root, path, result); return result; } };

为什么这次path同样是值传递 (无 &),但是也需要回溯,并需要回溯两次?

虽然这个代码也是值传递,但是在每一层path会执行“path += ”->", 并且将path传递给下一层,因此当下一层执行结束后,需要回溯将 “-” 和 “>" 回溯,

为什么不需要回溯节点的值?

因为是值传递 (无 &),每层对于 “path += to_string(root->val);” 的操作是针对本层的path,不影响上一层path,只会影响下一层的传递,因此当本层执行完之后,返回上一层不需要回溯节点的值。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/15 11:18:41

深入解析Windows OLE高危漏洞CVE-2025-21298:零点击远程代码执行

CVE-2025-21298 – Microsoft Windows OLE零点击远程代码执行漏洞分析 &#x1f4cc; 项目概述 本分析报告详细剖析了Microsoft Windows OLE组件中的一个严重安全漏洞CVE-2025-21298。该漏洞存在于ole32.dll库的UtOlePresStmToContentsStm函数中&#xff0c;是一个双重释放内…

作者头像 李华
网站建设 2026/6/15 11:18:39

2026 年最值得推荐的 Linux 游戏发行版

在很长一段时间里,“Linux 不适合玩游戏”几乎是默认共识。驱动麻烦、兼容性差、配置复杂,让不少玩家望而却步。 但到了 2026 年,这种印象已经明显过时。 随着 Steam / Proton 的成熟、显卡驱动的持续改进,以及越来越多发行版对游戏场景进行针对性优化,Linux 上可运行的游…

作者头像 李华
网站建设 2026/6/15 12:18:58

2026网络安全入门全攻略:从零基础到高薪就业,附工具 + 资源包

2026 网络安全入门全攻略&#xff1a;从零基础到高薪就业&#xff0c;附工具 资源包 2026 年的网络安全行业&#xff0c;早已不是小众技术领域 —— 人才缺口突破 480 万&#xff0c;一线城市平均年薪超 28 万&#xff0c;甚至大专学历也能凭借实战技能拿到 15K 月薪。 无论…

作者头像 李华
网站建设 2026/6/15 11:24:07

Spring Batch

Spring Batch 核心结构 Spring Batch 是一个用于批量处理的框架&#xff0c;提供了强大且灵活的功能&#xff0c;如事务管理、作业执行和数据分段处理等。其核心是基于 Job 和 Step 构建的。 Job 和 Step 的定义 Job&#xff1a;整个批处理作业的入口&#xff0c;可以包含多…

作者头像 李华
网站建设 2026/6/15 13:22:44

必收藏!程序员必看:别慌卷大模型,现有技术+AI才是破局关键

说真的&#xff0c;这两年混迹程序员圈子&#xff0c;看着身边一群搞技术的同行纷纷转向大模型赛道&#xff0c;心里挺有感触的。不管是深耕**Java、C**的后端开发者&#xff0c;专注前端页面搭建的前端工程师&#xff0c;还是做数据处理、架构设计的从业者&#xff0c;大家最初…

作者头像 李华
网站建设 2026/6/15 15:47:04

NTAI02模拟输入终端单元

NTAI02 模拟输入终端单元简介NTAI02 模拟输入终端单元用于工业控制系统中&#xff0c;将现场模拟信号转换为控制系统可处理的数据&#xff0c;实现精准监测与控制。支持多通道模拟信号输入提供高精度信号采集能力输入响应速度快&#xff0c;数据更新及时内置信号滤波与抗干扰设…

作者头像 李华