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允许原始类型的集合而不是要求集合中的元素成为完整的引用对象的能力特别感兴趣。 我在这篇文章中会更…

51nod1244 欧拉函数之和 杜教筛

和上一题差不多&#xff0c;一个是μ*Ie&#xff0c;一个是φ*IId 稍改就得到了这题的代码 &#xff08;我会告诉你我一开始逆元算错了吗&#xff09; 1 #include <bits/stdc.h>2 #define MAX 50000003 #define MOD 10000000074 using namespace std;5 long long a,b,N;6…

java7代码示例_Java中的七种排序方式代码示例

packagebaseJava;/*** title SortMethods.java*authorDonsenChen* Date 2018年5月2日 上午10:16:03* Description*/public classSortMethods {public static voidmain(String[] args) {int[] arr { 3, 7, 9, 1, 4, 8, 2, 6, 5};binarySort(arr);bubbleSort(arr);quickSort(arr…

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

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

android-DNS服务找不到

1、重启eclipse 2、重新建立AVD 3、在建立AVD时sd卡数值不要填转载于:https://www.cnblogs.com/shouhutian/p/6838327.html

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

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

与Selenium的集成测试

总览 我已经使用了一段时间&#xff0c;并且遇到了一些似乎可以使生活更轻松的事情。 我以为可以将其作为教程分享&#xff0c;所以我将向您介绍这些部分&#xff1a; 使用Maven设置Web项目&#xff0c;配置Selenium以在CI上作为集成测试运行 研究使用“页面对象”为网站中的…

MFC程序打开文件对话框出错的问题解决

前几天从网上下了个图像分析的mfc小程序&#xff0c;是VC6的 用VC6在本地编译生成都没问题。执行起来弹出一个未处理的错误&#xff0c;程序崩溃退出。 想起来原来遇到过打开文件对话框方面的问题&#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类&…

Linux命令之stty

用途说明 stty命令用于显示和修改终端行设置&#xff08;change and print terminal line settings&#xff09;。 常用参数 stty命令不带参数可以打印终端行设置&#xff0c;加上-a参数可以打印得更详细些。 stty size可以显示终端的大小&#xff0c;即行数和列数。 stty命令还…

本机速度文件支持的“纯” Java大数据存储

动机 所有这一切始于意识到我买不起足够大的计算机。 音频处理需要大量的内存。 Audacity是一款出色的免费音频处理器&#xff0c;它使用文件支持的存储系统对其进行管理。 这是解决此类问题的常用方法&#xff0c;在这些问题中&#xff0c;我们存储了大量信息&#xff0c;并希…

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

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

mysql 唯一记录_mysql选择唯一记录

我有以下查询&#xff1a;select * from members,subscriptionswhere members.MemberID subscriptions.MemberIDand subscriptions.Year 2009and members.ASSCID 15and subscriptions.Untildate between $2009-01-01 and 2009-12-31order by members.Memberlastname会员支付…

JAVA第七次作业

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

java时间格式转js_使用jquery或java脚本将日期时间转换为rfc3339格式

您的要求似乎微不足道&#xff0c;还有更多吗&#xff1f;这是显而易见的答案&#xff1a;function formatTimestring(s) {var b s.split(/[\\/:]/);return b[2] b[1] b[0] \T\ b[3] b[4] \00\ \Z\}alert(formatTimestring(\08/09/2010:12:00\) //20100908T120000Z);如…

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

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

asp.net FileUpload上传文件夹并检测所有子文件

1.在FileUpload控件添加一个属性 webkitdirectory""就可以上传文件夹了 <asp:FileUpload ID"FileUpload1" runat"server" webkitdirectory"" />2.检测文件夹下所有子文件 string DirectoryName FileUpload1.PostedFile.FileNam…

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

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

火狐配置java_java selenium+firefox环境搭建

已经成功搭建的版本关系&#xff1a;FireFox45selenium3.141.59 geckodriver 0.21.0启动浏览器后空白页&#xff1a;浏览器版本太高Firefox历史版本geckodriver驱动版本https://github.com/mozilla/geckodriver/releasesselenium maven地址org.seleniumhq.seleniumselenium-ja…