mysql递归查询所有上下节点_非递归打印二叉树的所有路径,保存父节点和孩子节点到底有啥差别...

题目解读

题目要求输出二叉树的所有路径(字符串形式),乍一看很简单,不就是二叉树的遍历嘛!其实不然,首先,我们用非递归的方式(C++)解决这道题(递归在产品代码中是不允许使用的,其次定位 bug 的时候非常困难)。这道题并非简单的 dfs(深度优先搜索),需要点技巧。

537ae31e7b9d0266db826d25d0003c9e.png

题目描述

标准 dfs 遍历输出 - 每次输出孩子节点

/**

* Definition for a binary tree node.

* struct TreeNode {

* int val;

* TreeNode *left;

* TreeNode *right;

* TreeNode(int x) : val(x), left(NULL), right(NULL) {}

* };

*/

class Solution {

public:

vector binaryTreePaths(TreeNode* root) {

if (root == nullptr) {

return {};

}

vector ans;

vector> allPath;

stack s;

vector path;

s.push(root);

path.push_back(root->val);

cout << path.back() << endl;

while (!s.empty()) {

TreeNode* curr = s.top();

s.pop();

if (curr->right != nullptr) {

path.push_back(curr->right->val);

s.push(curr->right);

cout << path.back() << endl;

}

if (curr->left != nullptr) {

path.push_back(curr->left->val);

s.push(curr->left);

cout << path.back() << endl;

}

}

return ans;

}

};

借助栈,上述代码实现了最简单的 dfs 遍历,结果如下:

2c324d8edf11620848f7f9e9fdfd7e37.png

标准 dfs 输出 - 每次输出孩子节点

栈中之所以先添加右叶子节点,是为了保证最终路径输出顺序从左往右。

标准 dfs 遍历输出 - 每次输出父亲节点

上述代码实现了标准 dfs 遍历,但显然不能满足题目要求,那该怎么处理呢?我们继续往下看。

class Solution {

public:

vector binaryTreePaths(TreeNode* root) {

if (root == nullptr) {

return {};

}

vector ans;

vector> allPath;

stack s;

vector path;

s.push(root);

path.push_back(root->val);

// cout << path.back() << endl;

while (!s.empty()) {

TreeNode* curr = s.top();

cout << curr->val << endl;

s.pop();

if (curr->right != nullptr) {

path.push_back(curr->right->val);

s.push(curr->right);

// cout << path.back() << endl;

}

if (curr->left != nullptr) {

path.push_back(curr->left->val);

s.push(curr->left);

// cout << path.back() << endl;

}

}

return ans;

}

};

上述代码中,我们打印栈顶每次弹出的元素,可以得到如下结果:

9a3e0ec02e774f185588a29969583794.png

标准 dfs 输出 - 每次输出父节点

也就是说,每次只要某个节点的孩子节点都入栈了,该节点就可以不用考虑了。这个例子中,栈中初始节点是 1,然后加入其孩子节点 3、2,1 出栈;再加入节点 2 的孩子节点 5,2 出栈;以此类推。可是,这还无法得到题目要求的结果呀!不着急,我们离答案已经越来越近了。

二叉树的所有路径

通过观察发现:每次从栈顶弹出的节点,如果它没有孩子节点,那么这个节点就是叶子节点,而且遇到的第一个叶子节点也是最左边的叶子节点。我们把每次从栈顶弹出的节点记录下来,保存在 vector 中,那么遇到第一个叶子节点时,我们就得到最左边的一条路径(1->2->5).但是,接下来栈顶弹出的元素可是 3,想要得到一条完整的路径,必须将 3 和 1->2 拼接。或者说,要知道 3 之前弹出了哪些元素,我们把 3 之前栈顶弹出的元素 和 3 拼接,再与 3 之后的路径拼接起来,即可得到第二条路径。如何实现这点是这道题的关键所在。很容易想到的一点是,在输出的前一条路径(vector)基础上,不断将尾部元素丢弃,那什么时候停止丢弃呢?显然,丢弃直到第一条路径的尾部元素的右孩子节点是 3 即可。所以,path 这个 vector 中就不能记录节点的值了,而要记录节点的地址,因为值不是唯一的,而地址是唯一的。

/**

* Definition for a binary tree node.

* struct TreeNode {

* int val;

* TreeNode *left;

* TreeNode *right;

* TreeNode(int x) : val(x), left(NULL), right(NULL) {}

* };

*/

class Solution {

public:

vector binaryTreePaths(TreeNode* root) {

if (root == nullptr) {

return {};

}

vector ans;

stack s;

vector path;

s.push(root);

string tmp;

while (!s.empty()) {

TreeNode* curr = s.top();

s.pop();

path.push_back(curr);

if (curr->right != nullptr) {

s.push(curr->right);

}

if (curr->left != nullptr) {

s.push(curr->left);

}

if (curr->left == nullptr && curr->right == nullptr) {

for (auto node : path) {

tmp += to_string(node->val);

if (node != path.back()) {

tmp += "->";

}

}

ans.push_back(tmp);

tmp.clear();

while (path.empty() == false && s.empty() == false && path.back()->right != s.top()) {

path.pop_back();

}

}

}

return ans;

}

};

这一次,终于得到了我们想要的结果了。上述代码中记录 string 的过程可以有更有效的方法,这里为了使代码的呈现更加容易理解,就不处理了,读者可以自己去研究下。

521f8fbb0ab6a975ba5174b06bb73d54.png

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

发现大量Java原语集合处理

在阅读博客文章5减少Java垃圾收集开销的技巧时 &#xff0c;我想起了一个名为Trove的小型Java收集库&#xff0c;该库“为Java提供了高速的常规和原始收集”。 我对应用Trove允许原始类型的集合而不是要求集合中的元素成为完整的引用对象的能力特别感兴趣。 我在这篇文章中会更…

nginx配置多个server_Nginx基本属性配置详解

. Nginx服务的基本配置1.1 用于调试进程和定位问题的配置项是否以守护进程的方式运行nginx# 默认ondaemon on|off;是否以master/worker方式工作# 默认on&#xff0c;指定了是否以master-worker进程的方式运行&#xff0c;如果设置为off&#xff0c;那么所有的请求将只会由maste…

数据增长率怎么算_20年老股民告诉你5个数据可轻松算出股价是否高估

自从开通自媒体以来&#xff0c;有很多朋友问我该如何正确判断股票的合理价格呢&#xff1f;说得太专业&#xff0c;可能很多新入市的股民朋友看不懂。那有没有一种估值方法通俗易懂呢&#xff1f;说实话&#xff0c;真有点为难我了。我首先想到了最简单的PE估值法&#xff0c;…

ad 单点登录 java 访问权限_AD 单点登录以及windows认证详细说明

上篇博客我谈到了一些关于ASP.NET Forms身份认证方面的话题&#xff0c;这次的博客将主要介绍ASP.NET Windows身份认证。Forms身份认证虽然使用广泛&#xff0c;不过&#xff0c;如果是在 Windows Active Directory 的环境中使用ASP.NET&#xff0c; 那么使用Windows身份认证也…

index加载显示servlet数据_[WEB篇]-JavaWeb基础与应用-02-Servlet开发

JavaWeb基础与应用2.Servlet开发Servlet是sun公司提供的一门用于开发动态web资源的技术。Sun公司在其API中提供了一个servlet接口&#xff0c;用户若想用发一个动态web资源(即开发一个Java程序向浏览器输出数据)&#xff0c;需要完成以下2个步骤&#xff1a;编写一个Java类&…

element ui后台html_GitHub上10个开源且优秀的后台管理系统UI面板

作者&#xff1a;SevDotwww.jianshu.com/p/3bc7404af887Web 开发中几乎的平台都需要一个后台管理&#xff0c;但是从零开发一套后台控制面板并不容易&#xff0c;幸运的是有很多开源免费的后台控制面板可以给开发者使用&#xff0c;那么有哪些优秀的开源免费的控制面板呢&#…

JAVA第七次作业

《Java技术》第七次作业 &#xff08;一&#xff09;学习总结 1.写出事件处理模型中的几个关键词&#xff0c;并通过具体代码实例说明你对事件处理模型的理解。 WindowListener&#xff1a;窗体事件&#xff0c;专门处理窗体的事件监听口&#xff0c;窗体的所有变化都可以使用此…

输出以下图案菱形7行_春夏格子图案超流行,三木的一款格子连衣裙,带来田园少女风...

春夏搭配中&#xff0c;增添了华丽格子图案搭配&#xff0c;从经典的格子裙子&#xff0c;到衬衫裙等。根据搭配不同而成为不同风格。所以&#xff0c;这一次&#xff0c;重点介绍格子裙和长衬衫的几种种搭配。格子裙子的春夏搭配推荐LOOK&#xff1a;1 [格子褶皱裙子棕色T恤]的…

使用Project Jigsaw的JDK 9 Early Access上的Eclipse IDE

几周前&#xff0c;我写了关于在Java 9上运行Eclipse Neon的文章 &#xff08;尽管&#xff0c;我在帖子标题中错误地和令人尴尬地留下了“火星”&#xff09;。 值得注意的是&#xff0c;我列出的步骤也适用于带有Project Jigsaw &#xff08;Java模块化&#xff09;构建的JDK…

arduino蜂鸣器_板卡推荐BPIUNO32 arduino 开发板,支持webduino与arduino应用

BPI:UNO32(也称为BPI-UNO32&#xff0c;被称为BPI UNO32)是一个带有Xtensa 32位LX6的单/双核心处理器的嵌入式系统的ESP32。支持Webduino和arduino的功能。BPI-UNO32使用的是esp-WROOM32&#xff0c;MCU。ESP32是一种集成2.4 GHz Wi-Fi和蓝牙双模式的单芯片解决方案。该公司的4…

Unity优化之GC——合理优化Unity的GC (难度3 推荐5)

原文链接&#xff1a;http://www.cnblogs.com/zblade/p/6445578.html 最近有点繁忙&#xff0c;白天干活晚上抽空写点翻译&#xff0c;还要运动&#xff0c;所以翻译工作进行的有点缓慢 。 本文续接前面的unity的渲染优化&#xff0c;进一步翻译Unity中的GC优化&#xff0c;英文…

centos重置系统_双系统下Linux系统无法启动及其引导丢失之解决

背景介绍: 很久很久以前, 我在 NewSurfacePro(SP5) 里插了一张 128G 内存卡, 费力九牛二虎之力在上面装了 Deepin, 后来在某次不知道是 Windows 还是 Deepin 更新后, Deepin 启动时总要发生一个极具 Linux 特色的启动错误, witch 似乎在我树莓派上出现过, 折腾了很久也没好, 就…

Java 7和Java 8之间的细微自动关闭合同更改

Java 7的try-with-resources语句和与该语句一起使用的AutoCloseable类型的一个不错的功能是&#xff0c;静态代码分析工具可以检测到资源泄漏。 例如&#xff0c;Eclipse&#xff1a; 具有以上配置并尝试运行以下程序时&#xff0c;您将收到三个警告&#xff1a; public stat…

reduce python3_更少循环?看看这3个Python函数

原标题&#xff1a;更少循环&#xff1f;看看这3个Python函数 全文共1146字&#xff0c;预计学习时长5分钟图源&#xff1a;wired 诞生于1991年的Python&#xff0c;这几年突然火了。简历上有了Python&#xff0c;就业竞争力瞬间提升&#xff0c;甚至一些小学教材上都出现了Pyt…

java图片上传(mvc)

最近有开始学起了java,好久没写文章了,好久没来博客园了。最近看了看博客园上次写的图片上传有很多人看&#xff0c;今天在一些篇关于java图片上传的。后台接收用的是mvc。不墨迹了&#xff0c;直接上图。 先看目录结构。idea开发。 一、图片上传还是使用的这款jq插件。前端部署…

appengine_Google AppEngine:任务队列API

appengine任务队列 com.google.appengine.api.taskqueue 使用任务队列&#xff0c;用户可以发起一个请求&#xff0c;以使应用程序执行此请求之外的工作。 它们是进行后台工作的强大工具。 此外&#xff0c;您可以将工作组织成小的离散单元&#xff08;任务&#xff09;。 然后…

Android studio Error occurred during initialization of VM 问题解决

最近开发导入其他Android项目遇见的问题,如下图&#xff1a; 解决办法&#xff1a; 将org.gradle.jvmargs的值该为521&#xff08;堆内存分配过高导致&#xff09; 备忘&#xff0c;希望能帮助到大家转载于:https://www.cnblogs.com/yunfang/p/6857096.html

滑动窗口限流 java_Spring Boot 的接口限流算法优缺点深度分析

点击上方蓝色字体&#xff0c;选择“标星公众号”优质文章&#xff0c;第一时间送达上一篇&#xff1a;这300G的Java资料是我师傅当年给我的&#xff0c;免费分享给大家(已修复)下一篇&#xff1a;昨天分享资料不小心把百度网盘深处的秘密泄露了(已修复)转自: loubobooo原文&am…

mips j指令_MIPS的基本实现

MIPS核心指令集&#xff1a;指令集概括为3个指令类&#xff1a;存储器访问指令类lw&#xff0c;sw等2. 算术逻辑指令类add&#xff0c;sub等3. 转移指令类beg&#xff0c;J等指令的共同性取指令&#xff0c;送PC值给Memory&#xff1b;根据指令内容读取寄存器内容&#xff1b;除…

腾讯管家去除桌面快捷小图标

找了大半天&#xff0c;原来腾讯管家也可以&#xff1a; 废话不多说&#xff0c;动手实践吧。。。。。 打开电脑管家——电脑诊所——桌面图标——【去掉快捷方式小箭头】 就这样轻松的实现了&#xff0c;赶快试试吧&#xff01; 转载于:https://www.cnblogs.com/yjq520/p/6858…