623. Add One Row to Tree
Problem Description
The goal of this problem is to add a new row to a binary tree, with all new nodes having the same value. The new row should be added at a specified depth in the tree, where the root node starts at depth 1. Here's how the process should work:
- If
depth
equals 1, a new root node with the givenval
should be created, and the entire original tree becomes the left subtree of this new root. - For depths greater than 1, we look for all nodes at
depth - 1
. For each of these nodes, we create two new children with the givenval
.- The new left child becomes the parent of the original left subtree of the node.
- The new right child becomes the parent of the original right subtree of the node.
- This process effectively inserts a row of new nodes at the specified
depth
, pushing the existing nodes at that depth (if any) to become children of the newly added nodes.
Intuition
The intuition behind the solution is to traverse the tree and locate the nodes at depth - 1
. For each of these nodes, we then attach new children nodes with the given val
. The essential steps we follow are:
- If
depth
is 1, we don't need to traverse the tree, because we simply create a new root with the givenval
and link the entire tree as the left subtree of this new root node. - If
depth
is greater than 1, we use depth-first traversal (DFS) to reach the nodes atdepth - 1
.- During the traversal, we keep track of the current depth.
- Once we reach the required level (
depth - 1
), we perform the insertion of new nodes.- This involves creating two new tree nodes with
val
as their value. - The new left node takes the current node's left subtree, and the new right node takes the current node's right subtree.
- This involves creating two new tree nodes with
- After performing the insertions at the required depth, we ensure the rest of the tree remains unchanged by only applying changes where necessary.
In this approach, we modify the tree in place without creating a separate structure, and we only create new nodes where the row is supposed to be added.
Learn more about Tree, Depth-First Search, Breadth-First Search and Binary Tree patterns.
Solution Approach
The provided solution utilizes recursion for a depth-first search approach to solve the problem efficiently. Here's a step-by-step explanation of how the algorithm operates:
- The
dfs
function defined within theSolution
class recursively explores the binary tree. - The
dfs
function takes two parameters:root
which represents the current node in the binary tree andd
which indicates the current depth of the recursive call. - The base case checks if
root
isNone
, in which case the function simply returns without performing any action, as we've reached a leaf node's child. - If the current depth
d
is equal todepth - 1
, it means we've reached the level above where the new row should be inserted. We perform the following insertions in this case:- Create a new
TreeNode
with a value ofval
and set its left child to the current node's original left subtree (root.left
). The new node is then assigned toroot.left
. - Similarly, create another new
TreeNode
with a value ofval
for the right side and assign the current node's original right subtree (root.right
) to the new node's right child. This new node is then assigned toroot.right
.
- Create a new
- If the current depth
d
is not yet atdepth - 1
, the function makes recursive calls to continue the search down the left and right subtrees, respectively, incrementing the depthd
by 1. - The main part of the
addOneRow
method checks ifdepth
equals 1. If so, a newTreeNode
is created with the specifiedval
and the entire original tree as its left subtree. This new node becomes the new root. - If
depth
is greater than 1, the recursivedfs
call is initiated withroot
and a starting depth of 1. - After the recursive calls complete, the original
root
of the tree is returned with the modifications in place (unless a new root was created, in which case that is returned).
The algorithm effectively leverages the call stack as its primary data structure, storing the state of each node's exploration during the recursion. The overall time complexity of this solution is O(n), where n is the number of nodes in the tree, since in the worst case, every node is visited once.
Ready to land your dream job?
Unlock your dream job with a 2-minute evaluator for a personalized learning plan!
Start EvaluatorExample Walkthrough
To illustrate the solution approach, let's consider a binary tree and the task of adding a row of nodes with value v
at a given depth k
.
Assume we have the following binary tree:
1 4 2 / \ 3 2 6 4 / \ / \ 53 1 5 7
And we want to add a row of nodes with value 5
at depth 3
.
Starting with addOneRow
, we check if the depth is 1. It's not, since we want to add the row at depth 3
. Therefore, we proceed to call the dfs
function passing the root of the tree and the initial depth 1
.
The dfs
function begins to traverse the tree. At the initial depth, none of the conditions to insert a node are met, so the function recursively calls itself for the left child 2
and right child 6
of the root 4
, with depth 2
.
For both child nodes, 2
and 6
, we are still not at the target depth (depth - 1
which is 2
) for insertion, so the function recursively calls itself for their children, with depth incremented to 3
, which is our target for insertion.
Node 2
has children 3
and 1
, and node 6
has children 5
and 7
. Now that d
equals depth - 1
at this level, we perform the insertions:
- The original left child of node
2
(which is3
) becomes the left child of a new node withval
5
, and this new node is then assigned as the new left child of node2
. - Similarly, the original right child of node
2
(which is1
) becomes the left child of another new node withval
5
, which is then assigned as the new right child of node2
.
The same process occurs for node 6
and its children.
With insertions complete, the function returns to its caller at the higher level and ultimately back to addOneRow
, which returns the original root of the binary tree.
After the row addition, the modified binary tree looks like this:
1 4 2 / \ 3 2 6 4 / \ / \ 55 5 5 5 6/ / \ 73 1 7 8 \ 9 5
In this example, only the nodes that needed new children were changedโ2
and 6
โand no other part of the tree was modified unnecessarily.
Solution Implementation
1# Definition for a binary tree node.
2class TreeNode:
3 def __init__(self, value=0, left=None, right=None):
4 self.value = value
5 self.left = left
6 self.right = right
7
8class Solution:
9 def addOneRow(self, root: Optional[TreeNode], value: int, depth: int) -> Optional[TreeNode]:
10 # Helper function to perform Depth-First Search (DFS) on the binary tree
11 def depth_first_search(node, current_depth):
12 # If node is None (the base case), we have reached a leaf's child and we return
13 if node is None:
14 return
15
16 # If we have reached the desired depth, we add the new row with value
17 if current_depth == depth - 1:
18 # Create new nodes with the given value and link to the previous children
19 node.left = TreeNode(value, left=node.left, right=None)
20 node.right = TreeNode(value, left=None, right=node.right)
21 return
22
23 # Recursively call DFS on the left and right children, incrementing the depth
24 depth_first_search(node.left, current_depth + 1)
25 depth_first_search(node.right, current_depth + 1)
26
27 # Special case when the new row needs to be added at the root
28 if depth == 1:
29 # Create a new root with the given value and set the original root as its left child
30 return TreeNode(value, left=root)
31
32 # Begin DFS with the original root at the starting depth of 1
33 depth_first_search(root, 1)
34 return root
35
1// Definition for a binary tree node.
2class TreeNode {
3 int val;
4 TreeNode left;
5 TreeNode right;
6
7 TreeNode() {}
8
9 TreeNode(int val) { this.val = val; }
10
11 TreeNode(int val, TreeNode left, TreeNode right) {
12 this.val = val;
13 this.left = left;
14 this.right = right;
15 }
16}
17
18class Solution {
19 // Instance variables to store the value to be added and the target depth
20 private int value;
21 private int targetDepth;
22
23 // Main method to add a new row to the tree
24 public TreeNode addOneRow(TreeNode root, int value, int depth) {
25 // Handling the special case where the new row is to be added as the new root
26 if (depth == 1) {
27 return new TreeNode(value, root, null);
28 }
29 // Initialize the instance variables
30 this.value = value;
31 this.targetDepth = depth;
32 // Start the depth-first search (DFS) from the root
33 depthFirstSearch(root, 1);
34 return root;
35 }
36
37 // Helper method to perform depth-first search
38 private void depthFirstSearch(TreeNode node, int currentDepth) {
39 // If the node is null, there is nothing to do; return immediately
40 if (node == null) {
41 return;
42 }
43 // Check if we reached the parent level of the target depth
44 if (currentDepth == targetDepth - 1) {
45 // Create new nodes with the given value and make them children of the current node
46 TreeNode leftChild = new TreeNode(value, node.left, null);
47 TreeNode rightChild = new TreeNode(value, null, node.right);
48 // Update the current node's children to the newly created nodes
49 node.left = leftChild;
50 node.right = rightChild;
51 // No need to traverse further as we have added the row at the target depth
52 return;
53 }
54 // Recursively search the left and right subtrees, increasing the depth by 1
55 depthFirstSearch(node.left, currentDepth + 1);
56 depthFirstSearch(node.right, currentDepth + 1);
57 }
58}
59
1/**
2 * Definition for a binary tree node.
3 */
4struct TreeNode {
5 int val;
6 TreeNode *left;
7 TreeNode *right;
8 TreeNode() : val(0), left(nullptr), right(nullptr) {}
9 TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
10 TreeNode(int v, TreeNode *leftNode, TreeNode *rightNode) : val(v), left(leftNode), right(rightNode) {}
11};
12
13class Solution {
14public:
15 // Funtion to add one row to the tree at a given depth with the given value
16 TreeNode* addOneRow(TreeNode* root, int value, int depth) {
17 // If the depth is 1, create a new node with the given value and make the existing tree its right child
18 if (depth == 1) {
19 return new TreeNode(value, root, nullptr);
20 }
21
22 // Set class variables to use in recursive calls
23 targetValue_ = value;
24 targetDepth_ = depth;
25
26 // Start the depth-first search (DFS)
27 depthFirstSearch(root, 1);
28
29 return root;
30 }
31
32private:
33 int targetValue_; // value to be added
34 int targetDepth_; // depth at which to add the new row
35
36 // Recursively traverse the tree to find the proper insertion depth
37 void depthFirstSearch(TreeNode* node, int currentDepth) {
38 // Base case: if the node is null, stop recursion
39 if (!node) {
40 return;
41 }
42
43 // When the target depth is reached, insert new nodes with targetValue_
44 if (currentDepth == targetDepth_ - 1) {
45 // Insert new left and right nodes between the current node and its children
46 TreeNode *newLeftNode = new TreeNode(targetValue_, node->left, nullptr);
47 TreeNode *newRightNode = new TreeNode(targetValue_, nullptr, node->right);
48
49 // Update the current node's children to point to the new nodes
50 node->left = newLeftNode;
51 node->right = newRightNode;
52
53 return; // No need to go deeper as we have already added the new row at this depth
54 }
55
56 // If not at the target depth yet, keep going deeper into the tree
57 depthFirstSearch(node->left, currentDepth + 1);
58 depthFirstSearch(node->right, currentDepth + 1);
59 }
60};
61
1// Function to add a new row at the given depth with the specified value in a binary tree.
2function addOneRow(root: TreeNode | null, val: number, depth: number): TreeNode | null {
3 // If the depth is 1, create a new root node with the current root as its left child.
4 if (depth === 1) {
5 return new TreeNode(val, root);
6 }
7
8 // Initialize a queue to perform level order traversal.
9 const queue: (TreeNode | null)[] = [root];
10
11 // Traverse the tree until the level before the desired depth is reached.
12 for (let currentDepth = 1; currentDepth < depth - 1; currentDepth++) {
13 const levelSize = queue.length; // Number of nodes at the current level.
14 for (let i = 0; i < levelSize; i++) {
15 // Remove the first node from the queue.
16 const currentNode = queue.shift();
17 // Add the left child to the queue if it exists.
18 if (currentNode?.left) queue.push(currentNode.left);
19 // Add the right child to the queue if it exists.
20 if (currentNode?.right) queue.push(currentNode.right);
21 }
22 }
23
24 // For each node at the target depth, add new nodes as their left and right children.
25 for (const parentNode of queue) {
26 if (parentNode) {
27 // Insert the new left child with the existing left child as its left subtree.
28 parentNode.left = new TreeNode(val, parentNode.left);
29 // Insert the new right child with the existing right child as its right subtree.
30 parentNode.right = new TreeNode(val, null, parentNode.right);
31 }
32 }
33
34 // Return the original root as the new tree with the added row.
35 return root;
36}
37
Time and Space Complexity
The provided code defines a solution to add a row of nodes with a specific value at a given depth in a binary tree. To analyze the time and space complexity, let's consider n
to be the total number of nodes in the binary tree.
Time Complexity:
The time complexity of the code can be determined by the number of nodes the algorithm visits. The function dfs
is a recursive function that visits each node exactly once when the depth is equal to or greater than the depth of insertion (d >= depth
).
In the worst-case scenario, which occurs when the new row is added at the maximum depth of the tree, the algorithm must visit all n
nodes to determine their depth and to potentially add the new nodes.
Therefore, the time complexity of the code is O(n)
.
Space Complexity:
Space complexity comes from the recursive stack space used in the depth-first search (DFS). In the worst-case scenario, the depth of the recursive call stack is proportional to the height of the tree.
- In the case of a balanced binary tree, the height of the tree is logarithmic, and the space complexity would be
O(log n)
. - For a skewed binary tree (a tree in which every node has only one child), the height could be as high as
n
, and the worst-case space complexity would beO(n)
.
Therefore, the overall space complexity is O(h)
where h
is the height of the tree, which ranges from O(log n)
for a balanced tree to O(n)
for a skewed tree.
Learn more about how to find time and space complexity quickly using problem constraints.
In a binary min heap, the maximum element can be found in:
Recommended Readings
Everything About Trees A tree is a type of graph data structure composed of nodes and edges Its main properties are It is acyclic doesn't contain any cycles There exists a path from the root to any node Has N 1 edges where N is the number of nodes in the tree and
https algomonster s3 us east 2 amazonaws com cover_photos dfs svg Depth First Search Prereqs Recursion Review problems recursion_intro Trees problems tree_intro With a solid understanding of recursion under our belts we are now ready to tackle one of the most useful techniques in coding interviews Depth First Search DFS
https algomonster s3 us east 2 amazonaws com cover_photos bfs svg Breadth First Search on Trees Hopefully by this time you've drunk enough DFS Kool Aid to understand its immense power and seen enough visualization to create a call stack in your mind Now let me introduce the companion spell
Got a question?ย Ask the Monster Assistantย anything you don't understand.
Still not clear? ย Submitย the part you don't understand to our editors. Or join ourย Discord and ask the community.