Leetcode算法【114. 二叉树展开为链表】
上周通过一位小伙伴,加入了一个氛围很好的小群,人不多,但是大家保持着对知识的渴望,让我很感动。
我自己也有一个群,人数也不多,但是能真正互动起来一起学习,一起进步的,还是太少。所以,现在也在学习如何让自己成为更好的群主,带动群活跃,带动一个社群活跃,带动小伙伴们一起进步,是我的愿景。当然,也不否认现在很多群友正在朝着积极向上的方向走着,我要做的,也是时刻保持对知识的渴望,做到“持续学习”。
谁让咱是一名优秀的程序员呢。上周日也学习了一遍递归,还通过一个二叉树的例子来简单介绍了下。我之前解决二叉树相关的问题,基本上用的都是递归,结果那天分享的朋友用了队列,让我眼前一亮,原来程序的世界真是奇妙。
所以,思想碰撞真的是一件很开心的事情。大家在持续的学习,持续的交流中,会打开一些思维定式,接纳更多的方式,你们觉得呢?
Algorithm LeetCode算法
114. 二叉树展开为链表
(https://leetcode-cn.com/problems/flatten-binary-tree-to-linked-list/
)
题目描述:给定一个二叉树,原地将它展开为链表。
例如,给定二叉树
示例1:
1
/ \
2 5
/ \ \
3 4 6
将其展开为:
1
\
2
\
3
\
4
\
5
\
6
本文题解参考地址:https://leetcode-cn.com/problems/flatten-binary-tree-to-linked-list/solution/xiang-xi-tong-su-de-si-lu-fen-xi-duo-jie-fa-by--26/
解法:后序遍历法
题目其实就是将二叉树通过右指针,组成一个链表。
从例子上可以看出,其实就是让我们把二叉树,通过先序遍历展示出来。所以我们首先想到的是能不能用先序遍历的方式,每遍历一个节点,就将上一个节点的右指针更新为当前节点。
先序遍历的顺序是 1->2->3->4->5->6,如下:
遍历到2
,把1
的右指针指向2
,即变成 1->2 3 4 5 6
遍历到3
,把2
的右指针指向3
,即变成 1->2->3 4 5 6
理想状况下,以此类推即可。
但是,如果我们把1
的右指针指向2
,那么这时候1
原本的右节点就丢失了,也就是我们后续找不到5
这个节点。
所以,又引起了我们的思考,如何才能不让5
丢失呢?后序遍历可以吗?
也就是我们依次遍历6 5 4 3 2 1
,然后每遍历一个节点就将当前节点的右指针更新为上一个节点,如下:
遍历到5
,把5
的右指针指向6
,即变成6 <- 5 4 3 2 1
遍历到4
,把4
的右指针指向5
,即变成6 <- 5 <- 4 3 2 1
以此类推,因为我们更新当前右指针的时候,当前节点的右节点已经访问过了,所以就不会存在丢失节点的问题。
把这个转变成后序遍历,遍历顺序就是 右子树 -> 左子树 -> 根节点
// 将二叉树构建完成
public static void main(String[] args) {
TreeNode treeNode = new TreeNode(1);
treeNode.left = new TreeNode(2);
treeNode.left.left = new TreeNode(3);
treeNode.left.right = new TreeNode(4);
treeNode.right = new TreeNode(5);
treeNode.right.right = new TreeNode(6);
flattern(treeNode);
}
/**
*
* @Title :
* @Description: 后续遍历
* @param treeNode
* @return :void
* @throws
*/
public static void flattern(TreeNode root) {
Stack<TreeNode> treeNodes = new Stack<>();
TreeNode current = root;
TreeNode preview = null;
while (current != null || !treeNodes.isEmpty()) {
while (current != null) {
// 添加根节点
treeNodes.push(current);
// 添加右节点
current = current.right;
}
// 已经访问到最右边的节点
current = treeNodes.peek();
// 当右节点已经被访问过或者左节点不存在的情况,就去访问根节点
if (current.left == null || current.left == preview) {
treeNodes.pop();
current.right = preview;
current.left = null;
preview = current;
current = null;
} else {
current = current.left;
}
}
}
补充说明:先序遍历
在介绍着后序遍历的时候,我们先用先序遍历的例子以及缺陷,来说明为什么我们选择后序遍历。那么,就一定不能用先序遍历了吗?显然,答案是不对的。
有一种特殊的先序遍历,提前将右节点保存到栈中,我们利用这种遍历方式就可以防止右节点的丢失。因为栈是先进后出,所以我们先将右节点入栈。
再根据上面先序遍历的分析,因为我们用栈保存了右孩子,所以不需要担心右孩子丢失了。用一个 pre 变量保存上次遍历的节点即可。
public static void flatten1(TreeNode root) {
if (root == null){
return;
}
Stack<TreeNode> s = new Stack<TreeNode>();
s.push(root);
TreeNode pre = null;
while (!s.isEmpty()) {
TreeNode temp = s.pop();
if(pre!=null){
pre.right = temp;
pre.left = null;
}
if (temp.right != null){
s.push(temp.right);
}
if (temp.left != null){
s.push(temp.left);
}
pre = temp;
}
}
结语
二叉树,是一颗神奇的树,理论上我们都可以通过先序、中序、后续遍历来拆解他,得到我们想要的结果,在做题的时候也是如此。
但是真正体现到编程的世界里,还是和理论做题有一点不同,编程需要我们用机器语言是实现,去思考,去打通我们算法的任督二脉,这样就是LeetCode存在的魅力,他是一个氛围很好的社区,拥有它,就拥有了算法的世界,拥有了进阶的可能。
安利了LeetCode,安利下自己的号,尽量做到每周一题,分模块的学习。小编最大的后悔就是没能在大学学好数据结构和算法这门课,现在吃亏了,吃大亏了。好在万变不离其宗,只要通过自己的勤奋,没有什么是不可能的。
加油,奔跑吧兄弟们!
作者:Dimple
公众号:奔跑吧攻城狮
本文由博客一文多发平台 OpenWrite 发布!
Leetcode算法【114. 二叉树展开为链表】的更多相关文章
- Java实现 LeetCode 114 二叉树展开为链表
114. 二叉树展开为链表 给定一个二叉树,原地将它展开为链表. 例如,给定二叉树 1 / \ 2 5 / \ \ 3 4 6 将其展开为: 1 \ 2 \ 3 \ 4 \ 5 \ 6 class S ...
- Leetcode 114.二叉树展开为链表
二叉树展开为链表 给定一个二叉树,原地将它展开为链表. 例如,给定二叉树 将其展开为: class Solution{ public: void flatten(TreeNode* root){ if ...
- [LeetCode] 114. 二叉树展开为链表 ☆☆☆(深度遍历)
二叉树展开为链表(很详细) 描述 给定一个二叉树,原地将它展开为链表. 例如,给定二叉树 1 / \ 2 5 / \ \3 4 6将其展开为: 1 \ 2 \ 3 \ 4 \ 5 \ 6 解析 变形的 ...
- leetcode 114. 二叉树展开为链表(Flatten Binary Tree to Linked List)
目录 题目描述: 示例: 解法: 题目描述: 给定一个二叉树,原地将它展开为链表. 示例: 给定二叉树 1 / \ 2 5 / \ \ 3 4 6 将其展开为: 1 \ 2 \ 3 \ 4 \ 5 \ ...
- LeetCode 114. 二叉树展开为链表(Flatten Binary Tree to Linked List)
题目描述 给定一个二叉树,原地将它展开为链表. 例如,给定二叉树 1 / \ 2 5 / \ \ 3 4 6 将其展开为: 1 \ 2 \ 3 \ 4 \ 5 \ 6 解题思路 二叉树转化为链表的基本 ...
- 【leetcode 114. 二叉树展开为链表】解题报告
思路:递归,将左子树变成单链表形式,再将右子树变成单链表形式,最后将左子树单链表的末端连接到右子树单链表表头,将根节点的左孩子置空 void flatten(TreeNode* root) { if ...
- LeetCode 114. 二叉树展开为链表 C++
/** * Definition for a binary tree node. * struct TreeNode { * int val; * TreeNode *left; * TreeNode ...
- [LeetCode] 114. Flatten Binary Tree to Linked List 将二叉树展开成链表
Given a binary tree, flatten it to a linked list in-place. For example,Given 1 / \ 2 5 / \ \ 3 4 6 T ...
- LeetCode(114): 二叉树展开为链表
Medium! 题目描述: 给定一个二叉树,原地将它展开为链表. 例如,给定二叉树 1 / \ 2 5 / \ \ 3 4 6 将其展开为: 1 \ 2 \ 3 \ 4 \ 5 \ 6 解题思路: 这 ...
随机推荐
- Unity的学习笔记(射线检测)
首先,射线检测的API是这样的,网上找了一下,这个图片看得很清楚: 接下来是自己使用这个进行测试 using System.Collections; using System.Collections. ...
- opencv::sift特征提取
SIFT特征检测介绍 SIFT(Scale-Invariant Feature Transform)特征检测关键特性: -建立尺度空间,寻找极值 -关键点定位(寻找关键点准确位置与删除弱边缘) -关键 ...
- libevent环境搭建
环境 VS2017.ubuntu16.04 官网网址:http://www.linuxfromscratch.org/blfs/view/svn/basicnet/libevent.html 下载 ...
- CSS布局解决方案(终结版)
作者:无悔铭 https://segmentfault.com/a/1190000013565024 前端布局非常重要的一环就是页面框架的搭建,也是最基础的一环.在页面框架的搭建之中,又有居中布局.多 ...
- linux 下ln命令--笔记
linux 下ln命令 ln命令用来为文件创建连接,连接类型分为硬连接和符号连接两种,默认的连接类型是硬连接.如果要创建符号连接必须使用"-s"选项.注意:符号链接文件不是一个独立 ...
- DOS批量重命名
下面有一些文件:dir /b *.*得到 144-10.1.4.150-1484707792127 2326-10.1.4.150-1484708376194 3682-10.1.4.150-1484 ...
- ThreadLocal小试牛刀
ThreadLocal中保存的数据只能被当前线程私有,不被其它线程可见 证明 声明一个全局的变量threadLocal,初始值为1,通过3个线程对其进行访问修改设置,理论上threadLocal的最终 ...
- 全方面讲解TensorFlow
任何曾经试图在 Python 中只利用 NumPy 编写神经网络代码的人都知道那是多么麻烦.编写一个简单的一层前馈网络的代码尚且需要 40 多行代码,当增加层数时,编写代码将会更加困难,执行时间也会更 ...
- 第三篇 Flask中的request
每个框架中都有处理请求的机制(request),但是每个框架的处理方式和机制是不同的,下面我们来了解一下Flask的request中都有什么东西 from flask import request 1 ...
- 匹马行天下之思维决定高度篇——道道道,学习Java之道
致Java星球的程序员兄弟们的一封信 亲爱的Java星球的程序员兄弟们: 你们好!我是来自地球的一名Java程序员,首先我代表地球人对贵星球的高司令来到地球传授Java语言,造福了全人类,造福了整个地 ...