-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy path116_populating-next-right-pointers-in-each-node.js
142 lines (133 loc) · 4.21 KB
/
116_populating-next-right-pointers-in-each-node.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
const { buildTreeBFS, TreeNode } = require('../_utils');
/**
*
* Problem:
* Given a binary tree, connect each node with its level order successor. The last
* node of each level should point to a null node.
* https://leetcode.com/problems/populating-next-right-pointers-in-each-node-ii/
*
*
* Time: O(n)
* Space: O(n)
*
* @param {TreeNode} root
* @return {TreeNode}
*/
function connectLevelNodeSibling(root) {
if (!root) {
return null;
}
const levelNodes = [root];
while (levelNodes.length > 0) {
let prevNode = null;
const levelLength = levelNodes.length;
for (let i = 0; i < levelLength; i++) {
const node = levelNodes.shift();
/**
* The key logic is here: maintain a prevNode, for the beginning of each
* level, set it to null, and update it when we traverse each node.
*/
if (prevNode) {
prevNode.next = node;
}
prevNode = node;
if (node.left) {
levelNodes.push(node.left);
}
if (node.right) {
levelNodes.push(node.right);
}
}
}
return root;
}
/**
* O(1) space solution which does not require storing all level nodes.
*/
function connectLevelNodeSiblingWithConstantSpace(root) {
let levelStart = root;
while (levelStart) {
/**
* A dummy node which represent the node before the first node of each level,
* so its next is always pointed to the first node of each level.
*/
const dummyNodeBeforeNextLevelStart = new TreeNode(-1);
/**
* This variable acts as a pointer to move `dummyNodeBeforeNextLevelStart`
* forward, who will connect the nodes on the next level. In each loop
* below, it will always point to the last child node of the `current`, so
* when `current` moves to next at the end of the loop, we can rely on this
* variable as the role of "prev" in the next loop.
*/
let nextLevelNode = dummyNodeBeforeNextLevelStart;
let current = levelStart;
while (current) {
if (current.left) {
nextLevelNode.next = current.left;
nextLevelNode = nextLevelNode.next;
}
if (current.right) {
nextLevelNode.next = current.right;
nextLevelNode = nextLevelNode.next;
}
current = current.next;
}
levelStart = dummyNodeBeforeNextLevelStart.next;
}
return root;
}
/**
* If the tree is a perfect binary tree (where each node must have 2 children expect
* the leaf nodes), the above solution can be simplified further.
* https://leetcode.com/problems/populating-next-right-pointers-in-each-node/
*
*/
function connectLevelNodeSiblingForPerfectBinaryTree(root) {
let levelStart = root;
while (levelStart) {
let current = levelStart;
while (current) {
if (current.left) {
// left node's next must be right node
current.left.next = current.right;
}
if (current.right && current.next) {
// right node's next must be next node's left
current.right.next = current.next.left;
}
current = current.next;
}
// go to next level, because left node is always existed expect leaf node
levelStart = levelStart.left;
}
return root;
}
// Test
const tree1 = buildTreeBFS([1, 2, 3, 4, 5, 6, 7]);
const result1 = connectLevelNodeSibling(tree1);
_printTreeLevelNodesWithNext(result1); // [1] [2, 3] [4, 5, 6, 7]
const tree2 = buildTreeBFS([12, 7, 1, null, 9, 10, 5]);
const result2 = connectLevelNodeSibling(tree2);
_printTreeLevelNodesWithNext(result2); // [12] [7, 1] [9, 10, 5]
const tree3 = buildTreeBFS([1, 2, 3, 4, 5, 6, 7]);
const result3 = connectLevelNodeSiblingForPerfectBinaryTree(tree3);
_printTreeLevelNodesWithNext(result3); // [1] [2, 3] [4, 5, 6, 7]
const tree4 = buildTreeBFS([12, 7, 1, null, 9, 10, 5]);
const result4 = connectLevelNodeSiblingWithConstantSpace(tree4);
_printTreeLevelNodesWithNext(result4); // [12] [7, 1] [9, 10, 5]
function _printTreeLevelNodesWithNext(root) {
let levelStart = root;
while (levelStart) {
const levelValues = [];
let current = levelStart;
levelStart = null;
while (current) {
levelValues.push(current.val);
if (!levelStart) {
levelStart = current.left || current.right;
}
current = current.next;
}
console.log(levelValues);
}
}