news 2026/2/16 13:55:18

二叉树--所有路径

作者头像

张小明

前端开发工程师

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/2/10 21:56:29

计算机毕业设计之基于springboot的流浪动物救助管理系统设计与实现

伴随着我国社会的发展&#xff0c;人民生活质量日益提高。于是对流浪动物救助管理进行规范而严格是十分有必要的&#xff0c;所以许许多多的信息管理系统应运而生。此时单靠人力应对这些事务就显得有些力不从心了。所以本论文将设计一套基于springboot的流浪动物救助管理系统&a…

作者头像 李华
网站建设 2026/2/13 23:41:49

如何科学选择软件开发公司:需求匹配与实力评估全解析

在数字化转型浪潮中&#xff0c;软件已成为企业运营与增长的核心引擎。无论是构建客户触达平台、优化内部流程&#xff0c;还是开发创新产品&#xff0c;选择一家靠谱的软件开发公司都是项目成功的关键第一步。然而&#xff0c;面对市场上数量众多、宣传各异的服务商&#xff0…

作者头像 李华
网站建设 2026/2/6 6:17:50

如何用AI预测软件缺陷?2026年智能测试术

AI驱动软件测试的新纪元 软件缺陷预测正从经验依赖转向数据智能驱动。2026年&#xff0c;AI技术通过机器学习&#xff08;ML&#xff09;、深度学习&#xff08;DL&#xff09;和自然语言处理&#xff08;NLP&#xff09;重塑测试范式&#xff0c;实现从“事后修复”到“事前预…

作者头像 李华
网站建设 2026/2/16 5:01:19

制作小商家线上引流工具,生成适配线上引流方案(朋友圈/短视频),提供文案模板,帮小商家快速获客,提高爆光。

1. 实际应用场景与痛点分析场景描述- 小商家&#xff08;餐饮店、服装店、美甲店、培训机构等&#xff09;想要通过线上渠道吸引顾客&#xff0c;但&#xff1a;1. 不懂如何写朋友圈或短视频文案。2. 不了解不同平台的引流玩法。3. 没有现成的模板&#xff0c;每次都要从零开始…

作者头像 李华
网站建设 2026/2/16 0:40:43

= delete和= default

你想弄明白C中 delete和 default这两个语法的区别与用法&#xff0c;它们是C11引入的核心特性&#xff0c;主要用于精准控制类的特殊成员函数&#xff08;如构造函数、拷贝构造、赋值运算符等&#xff09;。我会用通俗的语言实战示例&#xff0c;讲清两者的核心用途、区别和使用…

作者头像 李华