上周经过一位小伙伴,加入了一个氛围很好的小群,人很少,可是你们保持着对知识的渴望,让我很感动。java
我本身也有一个群,人数也很少,可是能真正互动起来一块儿学习,一块儿进步的,仍是太少。因此,如今也在学习如何让本身成为更好的群主,带动群活跃,带动一个社群活跃,带动小伙伴们一块儿进步,是个人愿景。固然,也不否定如今不少群友正在朝着积极向上的方向走着,我要作的,也是时刻保持对知识的渴望,作到“持续学习”。程序员
谁让咱是一名优秀的程序员呢。上周日也学习了一遍递归,还经过一个二叉树的例子来简单介绍了下。我以前解决二叉树相关的问题,基本上用的都是递归,结果那天分享的朋友用了队列,让我眼前一亮,原来程序的世界真是奇妙。算法
因此,思想碰撞真的是一件很开心的事情。你们在持续的学习,持续的交流中,会打开一些思惟定式,接纳更多的方式,大家以为呢?编程
114. 二叉树展开为链表
(https://leetcode-cn.com/problems/flatten-binary-tree-to-linked-list/
)数据结构
题目描述:给定一个二叉树,原地将它展开为链表。数据结构和算法
例如,给定二叉树学习
示例1:指针
1 / \ 2 5 / \ \ 3 4 6
将其展开为:code
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 发布!