0114-二叉树展开为链表
给你二叉树的根结点 root
,请你将它展开为一个单链表:
- 展开后的单链表应该同样使用
TreeNode
,其中right
子指针指向链表中下一个结点,而左子指针始终为null
。 - 展开后的单链表应该与二叉树 先序遍历 顺序相同。
示例 1:
**输入:** root = [1,2,5,3,4,null,6]
**输出:** [1,null,2,null,3,null,4,null,5,null,6]
示例 2:
**输入:** root = []
**输出:** []
示例 3:
**输入:** root = [0]
**输出:** [0]
提示:
- 树中结点数在范围
[0, 2000]
内 -100 <= Node.val <= 100
进阶: 你可以使用原地算法(O(1)
额外空间)展开这棵树吗?
解法一
可以发现展开的顺序其实就是二叉树的先序遍历。算法和 94 题中序遍历的 Morris 算法有些神似,我们需要两步完成这道题。
- 将左子树插入到右子树的地方
- 将原来的右子树接到左子树的最右边节点
- 考虑新的右子树的根节点,一直重复上边的过程,直到新的右子树为 null
可以看图理解下这个过程。
1 | 1 |
代码的话也很好写,首先我们需要找出左子树最右边的节点以便把右子树接过来。
1 | public void flatten(TreeNode root) { |
解法二
题目中,要求说是 in-place
,之前一直以为这个意思就是要求空间复杂度是 $O(1)$。偶然看见评论区大神的解释, in-place
的意思可能更多说的是直接在原来的节点上改变指向,空间复杂度并没有要求。所以这道题也可以用递归解一下。
1 | 1 |
利用递归的话,可能比解法一难理解一些。
题目其实就是将二叉树通过右指针,组成一个链表。
1 | 1 -> 2 -> 3 -> 4 -> 5 -> 6 |
我们知道题目给定的遍历顺序其实就是先序遍历的顺序,所以我们能不能利用先序遍历的代码,每遍历一个节点,就将上一个节点的右指针更新为当前节点。
先序遍历的顺序是 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
就找不到了。
解决方法的话,我们可以逆过来进行。
我们依次遍历 6 5 4 3 2 1
,然后每遍历一个节点就将当前节点的右指针更新为上一个节点。
遍历到 5
,把 5
的右指针指向 6
。6 <- 5 4 3 2 1
。
遍历到 4
,把 4
的右指针指向 5
。6 <- 5 <- 4 3 2 1
。
… …
1 | 1 |
这样就不会有丢失孩子的问题了,因为更新当前的右指针的时候,当前节点的右孩子已经访问过了。
而 6 5 4 3 2 1
的遍历顺序其实变形的后序遍历,遍历顺序是右子树->左子树->根节点。
先回想一下后序遍历的代码
1 | public void PrintBinaryTreeBacRecur(TreeNode<T> root){ |
这里的话,我们不再是打印根节点,而是利用一个全局变量 pre
,更新当前根节点的右指针为 pre
,左指针为 null
。
1 | private TreeNode pre = null; |
相应的左孩子也要置为 null
,同样的也不用担心左孩子丢失,因为是后序遍历,左孩子已经遍历过了。和 112 题一样,都巧妙的利用了后序遍历。
既然后序遍历这么有用,利用 112 题介绍的后序遍历的迭代方法,把这道题也改一下吧。
1 | public void flatten(TreeNode root) { |
解法三
解法二中提到如果用先序遍历的话,会丢失掉右孩子,除了用后序遍历,还有没有其他的方法避免这个问题。在 Discuss
又发现了一种解法。
为了更好的控制算法,所以我们用先序遍历迭代的形式,正常的先序遍历代码如下,
1 | public static void preOrderStack(TreeNode root) { |
还有一种特殊的先序遍历,提前将右孩子保存到栈中,我们利用这种遍历方式就可以防止右孩子的丢失了。由于栈是先进后出,所以我们先将右节点入栈。
1 | public static void preOrderStack(TreeNode root) { |
之前我们的思路如下:
题目其实就是将二叉树通过右指针,组成一个链表。
1 | 1 -> 2 -> 3 -> 4 -> 5 -> 6 |
我们知道题目给定的遍历顺序其实就是先序遍历的顺序,所以我们可以利用先序遍历的代码,每遍历一个节点,就将上一个节点的右指针更新为当前节点。
先序遍历的顺序是 1 2 3 4 5 6
。
遍历到 2
,把 1
的右指针指向 2
。1 -> 2 3 4 5 6
。
遍历到 3
,把 2
的右指针指向 3
。1 -> 2 -> 3 4 5 6
。
… …
因为我们用栈保存了右孩子,所以不需要担心右孩子丢失了。用一个 pre
变量保存上次遍历的节点。修改的代码如下:
1 | public void flatten(TreeNode root) { |
总结
解法一和解法三可以看作自顶向下的解决问题,解法二可以看作自底向上。以前觉得后序遍历比较麻烦,没想到竟然连续遇到了后序遍历的应用。先序遍历的两种方式自己也是第一次意识到,之前都是用的第一种正常的方式。