【leetcode刷题笔记】Populating Next Right Pointers in Each Node II
What if the given tree could be any binary tree? Would your previous solution still work?
Note:
- You may only use constant extra space.
For example,
Given the following binary tree,
1 / \ 2 3 / \ \ 4 5 7
After calling your function, the tree should look like:
1 -> NULL / \ 2 -> 3 -> NULL / \ \ 4-> 5 -> 7 -> NULL
最开始的想法是用一个队列和一个列表。当前遍历的层的元素放在队列中,在遍历当前层的时候,把下一层元素的next指针设置好,并且把下一层的元素放在列表中。之所以要用列表,是因为当遍历一个node的孩子的时候,它在next指针下的前驱就是列表中的最后一个元素,如果用队列的话就取不出最后一个元素,所以这里要用列表。在一层遍历结束后,要把列表中的元素全部放入到队列中,开始下一层的遍历,代码如下:
1 /** 2 * Definition for binary tree with next pointer. 3 * public class TreeLinkNode { 4 * int val; 5 * TreeLinkNode left, right, next; 6 * TreeLinkNode(int x) { val = x; } 7 * } 8 */ 9 public class Solution { 10 public void connect(TreeLinkNode root){ 11 if(root == null) 12 return; 13 14 Queue<TreeLinkNode> currLevel = new LinkedList<TreeLinkNode>(); 15 List<TreeLinkNode> nextLevel = new ArrayList<TreeLinkNode>(); 16 17 currLevel.add(root); 18 19 while(!currLevel.isEmpty()){ 20 while(!currLevel.isEmpty()){ 21 TreeLinkNode temp = currLevel.poll(); 22 if(temp.left != null){ 23 if(!nextLevel.isEmpty() && nextLevel.get(nextLevel.size()-1).next == null) 24 nextLevel.get(nextLevel.size()-1).next = temp.left; 25 nextLevel.add(temp.left); 26 } 27 if(temp.right != null){ 28 if(!nextLevel.isEmpty() && nextLevel.get(nextLevel.size()-1).next == null) 29 nextLevel.get(nextLevel.size()-1).next = temp.right; 30 nextLevel.add(temp.right); 31 } 32 } 33 List<TreeLinkNode> passby = new ArrayList<TreeLinkNode>(nextLevel); 34 currLevel.addAll(passby); 35 nextLevel.clear(); 36 } 37 } 38 }
但是题目中要求要常数的额外空间,上述代码居然也AC了,可见leetcode要求不是特别严格。
那么我们怎么把上述代码修改成原地算法呢?
首先,对于上述用到的列表,其实我们每次只需要列表的最后一个元素,它是当前遍历节点的孩子的前驱。如题目中所示的例子,在遍历2的右子5的时候,只要知道它的前驱是4,即刚刚遍历过的2的左子即可;同理,在遍历节点3的时候,要知道它的右孩子的前驱,只要知道刚刚遍历到的下一层节点5就可以了;所以我们只需要一个TreeNode prev记录刚刚通过父节点遍历到的下一层的节点就可以了。
其次,对于上述用到的队列,队列中元素的先后顺序其实我们在遍历上一层的时候确定这一层元素的next指针,那么在遍历这一层的时候就可以利用这个next指针了,但是仍然需要知道这一层的起点指针,这个起点指针也可以在遍历上一层的时候确定。
修改后的代码如下:
1 /** 2 * Definition for binary tree with next pointer. 3 * public class TreeLinkNode { 4 * int val; 5 * TreeLinkNode left, right, next; 6 * TreeLinkNode(int x) { val = x; } 7 * } 8 */ 9 public class Solution { 10 public void connect(TreeLinkNode root) { 11 TreeLinkNode first = root; 12 while(first != null){ 13 TreeLinkNode pointer = first; 14 first = null; 15 TreeLinkNode prev = null; 16 17 while(pointer != null){ 18 //更新first 19 if(pointer.left != null){ 20 if(first == null){ 21 first = pointer.left; 22 prev = first; 23 }else{ 24 prev.next = pointer.left; 25 prev = prev.next; 26 } 27 } 28 29 if(pointer.right != null){ 30 if(first == null){ 31 first = pointer.right; 32 prev = first; 33 }else{ 34 prev.next = pointer.right; 35 prev = prev.next; 36 } 37 } 38 39 pointer = pointer.next; 40 } 41 } 42 } 43 }
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
· 没有源码,如何修改代码逻辑?
· 全程不用写代码,我用AI程序员写了一个飞机大战
· DeepSeek 开源周回顾「GitHub 热点速览」
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 记一次.NET内存居高不下排查解决与启示
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了