Solution: Populating Next Right Pointers in Each Node
Let's solve the Populating Next Right Pointers in the Each Node problem using the Tree Breadth-first Search pattern.
Statement
Given a next
. This pointer is initially set to NULL
for all nodes. Your task is to connect all nodes of the same hierarchical level by setting the next
pointer to its immediate right node.
The next
pointer of the rightmost node at each level is set to NULL.
Constraints:
- The number of nodes in the tree is in the range .
-
Node.data
Solution
So far, you’ve probably brainstormed some approaches and have an idea of how to solve this problem. Let’s explore some of these approaches and figure out which one to follow based on considerations such as time complexity and any implementation constraints.
Naive approach
The naive approach to establishing connections between nodes at the same level in a binary tree is to utilize a queue data structure and perform a level order traversal. The steps of the algorithm are given below:
- Start with the root node of the binary tree.
- Create an empty queue data structure to perform the breadth-first traversal. Enqueue the root node into the queue.
- While the queue is not empty, perform the following steps:
- Initialize a variable,
levelSize
, to the current size of the queue (number of nodes in the current level). - Traverse through each node in the current level (equal to
levelSize
):- Dequeue the front node from the queue and call it the
currentNode
. - If
i
(current index in the level) is less thanlevelSize - 1
, set thenext
pointer ofcurrentNode
to the first node in the queue (front of the queue) representing the next node at the same level. - Enqueue the left and right children of
currentNode
(if they exist) into the queue.
- Dequeue the front node from the queue and call it the
- Initialize a variable,
- Return the root node of the modified binary tree with the connections between nodes using the
next
pointers.
The time complexity of this naive approach is , where is the number of nodes in the binary tree. This is because we visit each node once during the breadth-first traversal.
The space complexity of the naive approach is also . At any given time, the queue can store at most one-level nodes in the binary tree. The binary tree is a complete binary tree with nodes in the last level, which is proportional to . Therefore, the space complexity is .
Optimized approach using Tree Breadth-first Search
To establish connections between nodes at the same level in a binary tree, we can optimize the approach by leveraging the existing tree structure and employing a tree breadth-first search pattern. The goal is to create a linklist-like structure at each level, where each node points to its right sibling.
Instead of using an additional queue data structure, which consumes extra space, we can utilize the existing structure of the binary tree to achieve the same level-order traversal without the need for the queue.
The algorithm uses a nested while loop to traverse the tree and establish the connections:
- First, check if the root node is NULL. If it is, return NULL, indicating an empty tree.
- If the
root
is notNULL
, it initializesmostleft
as theroot
node. This variable keeps track of the leftmost node of the current level. - Iterate through each level of the tree. The outer loop continues until no more levels are below (
mostleft.left
is NULL).
- Inside the outer loop, there is an inner loop that traverses each node on the current level. It starts with the
current
variable initialized as the leftmost node of the current level. - Within the inner loop, the algorithm establishes the connections between nodes.
- First connection: It connects the
current
node’s left child to its right child by assigningcurrent.left.next
tocurrent.right
. - Second connection: If there is a next node (
current.next
) on the same level, it connects thecurrent
node’s right child to the left child of its next node by assigningcurrent.right.next
tocurrent.next.left
. - After connecting the nodes on the current level, the
current
moves to the next node on the same level. This process continues until the end of the level is reached (current
becomesNULL
).
- First connection: It connects the
- Once the inner loop completes, the
mostleft
is updated to the left child of the previousmostleft
node. This moves the traversal down to the next level. - The outer loop repeats the above steps until there are no more levels below (
mostleft.left
isNULL
). - Finally, the modified
root
node is returned.
Let’s look at the following illustration to get a better understanding of the solution:
Let’s look at the code for this solution below:
class ConnectSameLevelNodes {public static EduTreeNode<Integer> populateNextPointers(EduTreeNode<Integer> root) {if (root == null) {return null;}EduTreeNode<Integer> mostleft = root;while (mostleft.left != null) {EduTreeNode<Integer> current = mostleft;while (current != null) {current.left.next = current.right;if (current.next != null) {current.right.next = current.next.left;}current = current.next;}mostleft = mostleft.left;}return root;}// Driver codepublic static void main(String[] args) {// Initializing the binary treeList<EduTreeNode<Integer>> inputs = Arrays.asList(new EduTreeNode<Integer>(100), new EduTreeNode<Integer>(50), new EduTreeNode<Integer>(200), new EduTreeNode<Integer>(25), new EduTreeNode<Integer>(75), new EduTreeNode<Integer>(300), new EduTreeNode<Integer>(10));EduBinaryTree<Integer> tree = new EduBinaryTree<Integer>(inputs);// Function call to populate next pointersConnectSameLevelNodes.populateNextPointers(tree.root);int indexVal = 0;System.out.println("Binary tree:");Print.displayTree(tree.root);System.out.println();for (EduTreeNode<Integer> nodeValue : inputs) {indexVal++;// Function call to find current node and get its next node using next pointerEduTreeNode<Integer> tmp = tree.getNextNode(tree.root, nodeValue.data);String sibNode = "null";if (tmp != null) {sibNode = String.valueOf(tmp.data);}System.out.print(indexVal + ".");System.out.println("\tCurrent Node Value: " + nodeValue.data);System.out.println("\tNext Node Value: " + sibNode);System.out.println(new String(new char[100]).replace('\0', '-'));}}}
Solution summary
-
Check if the root node is empty. If it is, return NULL.
-
Traverse each level of the tree using an outer loop, starting from the leftmost node of the current level.
-
At each level, establish connections between nodes using an inner loop. Connect a node’s left child to its right child and connect a node’s right child to the left child of its next node on the same level.
-
Move to the next node on the same level, and after completing the level, update the leftmost node for the next level. Repeat the steps of connecting nodes at every level until all levels have been traversed.
-
Return the modified root node.
Time complexity
The time complexity of this solution is , where is the number of nodes in the binary tree.
Space complexity
The space complexity of this solution is , as we are not using any extra space.
Level up your interview prep. Join Educative to access 70+ hands-on prep courses.