1973. Count Nodes Equal to Sum of Descendants
Problem Description
In this problem, we're given the root of a binary tree. Our goal is to find how many nodes in this tree have values that are equal to the sum of the values of all its descendants. A descendant of a node x is any node that lies in the subtree rooted at x, including all the leaf nodes down the path from node x. If a node has no descendants (i.e., it is a leaf node), the sum of its descendants is considered to be 0. The task is to return the count of such nodes.
To simplify, let's say we have a node with the value 10 and it has two children with values 3 and 7, respectively. The sum of the values of the descendants is 3+7=10, which equals the node's value. This node should be counted as one. We need to do this for every node in the tree.
Intuition
The approach uses a classic tree traversal technique known as Depth-First Search (DFS). The DFS allows us to reach the leaf nodes and then go up the tree, summarizing the values as we go along. This particular solution is a post-order traversal, meaning we process a node after we've processed its children.
Here's the intuition behind the approach:
- Perform a DFS starting from the root.
- While traversing, calculate the sum of the values of the left and right subtrees individually.
- After obtaining the sum of the left and right subtree for a node, we check if the sum is equal to the node's value.
- If so, we increment a counting variable,
ans
, as we've found a node where the value equals the sum of its descendants. - After checking the condition, we return to the parent node the sum of the node's value plus all the values from its descendants.
- We incrementally build these sums from the child nodes up to the root, performing the checks at each step.
- The
nonlocal
keyword used inside thedfs
function is a way to modify theans
variable defined outside the nested function's scope.
The traversal ends when all nodes are visited, and the resulting ans
variable holds the count of nodes that meet the criteria. This is both an efficient and reliable way to determine the sum of descendants for each node in the tree since with post-order traversal, we visit all the children before the parent, ensuring we have all the necessary information when the sums are compared.
Learn more about Tree, Depth-First Search, Binary Search Tree and Binary Tree patterns.
Solution Approach
The solution leverages a recursive Depth-First Search (DFS) pattern to traverse the binary tree and calculate the sum of the descendants of each node while also keeping track of the count of nodes that fulfill the given criteria.
Here's a detailed breakdown of the implementation:
-
Recursive DFS Function (
dfs
): Thedfs
function is defined within theequalToDescendants
method. This function is a helper function that is called recursively to perform the post-order traversal.- It takes a single parameter, the current
root
of the subtree being processed. - Base Case: If we encounter a
None
node, it signifies that we have reached past a leaf node, and we return0
since there are no descendants to sum up. - Recursive Case: If the node is not
None
, we recursively calldfs
on the left and right children of the current node. - We sum up the results of
dfs
calls on the left and right children,l
andr
respectively, to get the total sum of the descendants. - We compare the sum of the descendant values
l + r
with the current node's valueroot.val
. If they are equal, we increment theans
counter. This is done using thenonlocal
keyword to modify theans
variable that is defined in the outer function's scope. - The
dfs
function returns the sum of the node's value and its descendants' values to its parent node. This isroot.val + l + r
.
- It takes a single parameter, the current
-
Recursive Traversal: The recursion stack unwinds, visiting all nodes from the bottom-up (post-order), ensuring each node is checked precisely once for the condition. This helps to optimize the function as the calculated sums for the subtrees are used directly without recalculating them.
-
Global Counter (
ans
): A counter namedans
is used to keep track of the number of nodes whose values equal the sum of their descendants' values. It is defined in the same scope as thedfs
function, but outside of it, to be accessible and modifiable from within the nesteddfs
function. -
Finalizing the Count: After initiating the traversal with
dfs(root)
, the functionequalToDescendants
ultimately returns the value ofans
, which is the total number of nodes that satisfy the condition at the end of the traversal.
The time complexity of the solution is O(N)
where N
is the number of nodes in the tree, as each node is visited exactly once. The space complexity is also O(N)
, which is the worst-case space used by the call stack during the DFS (this occurs in the case of a skewed tree).
In summary, by using a post-order DFS and a simple counter, we effectively check the sum of descendants for each node against its value and return the total count of such nodes.
Ready to land your dream job?
Unlock your dream job with a 2-minute evaluator for a personalized learning plan!
Start EvaluatorExample Walkthrough
Let's illustrate the solution approach with a small binary tree example. Consider the following binary tree:
1 10 2 / \ 3 4 6 4 / \ 5 3 1
We want to find out how many nodes have values equal to the sum of the values of their descendants.
Following the steps given in the solution approach:
- We start the DFS from the root node which is
10
. - We traverse to the left child
4
and then to it's left child3
which is a leaf node.- Leaf nodes are base cases, returning
0
since they have no descendants.
- Leaf nodes are base cases, returning
- Back to node
4
, we now traverse its right child which is1
and since it's a leaf node, it also returns0
. - Now we're back at
4
with the sum of its descendant being3 + 0 + 1 + 0 = 4
, which equals the node's value.- Therefore, we increment our counter
ans
by1
.
- Therefore, we increment our counter
- We return
4
(node's value) + the sum of its descendants4
, which is8
to the parent node10
. - We proceed to node
10
's right child6
, which is a leaf node.- Thus, it returns
0
as the sum of the descendants.
- Thus, it returns
- Now at the root node
10
, we take the sum of descendant values we got from the left8
and right0
child and compare it with the node10
's value.8 + 0
is not equal to10
, henceans
remains the same.
- The function
equalToDescendants
returns the finalans
value which is1
, so there is1
node with a value equal to the sum of its descendants.
In this example, the node with the value 4
is the only node meeting the criteria.
By following this post-order DFS approach, we calculate the sum of descendant nodes' values for each node and compare it to each node's own value, incrementing our counter when they match and finally returning the count of such nodes.
Solution Implementation
1class TreeNode:
2 # TreeNode class used to represent each node in a binary tree.
3 def __init__(self, val=0, left=None, right=None):
4 self.val = val # Value of the node
5 self.left = left # Left child of the node
6 self.right = right # Right child of the node
7
8
9class Solution:
10 def equalToDescendants(self, root: Optional[TreeNode]) -> int:
11 """
12 This method returns the count of nodes whose value is equal to the sum of values
13 of its descendants.
14
15 :param root: The root node of the binary tree.
16 :return: The count of nodes for which node's value == sum of descendants' values.
17 """
18
19 def sum_of_descendants(node):
20 """
21 This helper function calculates the sum of values of all descendants of the
22 current node, and updates the count if the sum equals the node's value.
23
24 :param node: The current node being processed.
25 :return: The sum of values including the current node and its descendants.
26 """
27 if node is None:
28 return 0
29
30 # Recursively compute the sum of values of left and right subtrees
31 left_sum = sum_of_descendants(node.left)
32 right_sum = sum_of_descendants(node.right)
33
34 # If sum of left and right children's values equals the current node's value, increment the count
35 if left_sum + right_sum == node.val:
36 nonlocal node_count
37 node_count += 1
38
39 # Return the sum of the current node's value and its descendants' values
40 return node.val + left_sum + right_sum
41
42 # Counter to hold the number of nodes meeting the specific condition.
43 node_count = 0
44
45 # Initiate depth-first search traversal from the root to calculate the sum of descendants and update the count.
46 sum_of_descendants(root)
47
48 # Return the final count of nodes whose value is equal to sum of their descendants.
49 return node_count
50
51
52# Additional code such as defining the TreeNode class, constructing a tree, and calling the Solution method
53# would be needed here to test the functionality of the code in practice.
54
1/**
2 * Definition for a binary tree node.
3 */
4class TreeNode {
5 int val;
6 TreeNode left;
7 TreeNode right;
8 TreeNode() {}
9 TreeNode(int val) { this.val = val; }
10 TreeNode(int val, TreeNode left, TreeNode right) {
11 this.val = val;
12 this.left = left;
13 this.right = right;
14 }
15}
16
17class Solution {
18 // Variable to keep track of the number of nodes equal to the sum of their descendants.
19 private int numNodesEqualToDescendants;
20
21 /**
22 * Public method that initiates the depth-first search traversal of the tree
23 * and returns the count of nodes where node value is equal to the sum of its descendants.
24 *
25 * @param root The root of the binary tree.
26 * @return The count of nodes where the node value is equal to the sum of its descendants.
27 */
28 public int equalToDescendants(TreeNode root) {
29 depthFirstSearch(root);
30 return numNodesEqualToDescendants;
31 }
32
33 /**
34 * Recursive helper method to perform the depth-first search and calculate the sum of the descendants.
35 *
36 * @param node The current node in the depth-first search traversal.
37 * @return The sum of this node and its descendant nodes.
38 */
39 private int depthFirstSearch(TreeNode node) {
40 // Base case: if the node is null, return 0 since it does not contribute to the sum.
41 if (node == null) {
42 return 0;
43 }
44
45 // Recursively compute the sum of the left and right subtrees.
46 int leftSum = depthFirstSearch(node.left);
47 int rightSum = depthFirstSearch(node.right);
48
49 // Check if the current node's value is equal to the sum of its left and right descendants.
50 if (leftSum + rightSum == node.val) {
51 // Increment the count if the current node's value matches the sum of its descendants.
52 numNodesEqualToDescendants++;
53 }
54
55 // Return the sum of the current node value and its descendants' values.
56 return node.val + leftSum + rightSum;
57 }
58}
59
1// Definition for a binary tree node.
2struct TreeNode {
3 int val;
4 TreeNode *left;
5 TreeNode *right;
6 TreeNode() : val(0), left(nullptr), right(nullptr) {}
7 TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
8 TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
9};
10
11class Solution {
12public:
13 // This function counts the number of nodes in a binary tree
14 // where the node value is equal to the sum of the values of its descendants.
15 int equalToDescendants(TreeNode* root) {
16 int count = 0; // Initialize a counter for the nodes that meet the condition.
17
18 // Define a Depth-First Search (DFS) lambda function that calculates the sum of a node's descendants
19 // and checks if the node's value equals that sum.
20 std::function<long long(TreeNode*)> dfs = [&](TreeNode* node) -> long long {
21 if (!node) {
22 return 0; // If the current node is null, return 0 as it doesn't contribute to the sum.
23 }
24
25 // Recursively call the dfs function on the left and right children and calculate their sums.
26 long long leftSum = dfs(node->left);
27 long long rightSum = dfs(node->right);
28
29 // If the current node's value is equal to the sum of its descendants, increment the count.
30 count += (leftSum + rightSum == node->val);
31
32 // Return the sum of the current node's value and its descendants' values.
33 return node->val + leftSum + rightSum;
34 };
35
36 // Start the DFS traversal from the root to count the nodes satisfying the condition.
37 dfs(root);
38
39 // Return the final count of nodes.
40 return count;
41 }
42};
43
1// Definition for a binary tree node in TypeScript.
2class TreeNode {
3 val: number;
4 left: TreeNode | null;
5 right: TreeNode | null;
6
7 constructor(val: number = 0, left: TreeNode | null = null, right: TreeNode | null = null) {
8 this.val = val;
9 this.left = left;
10 this.right = right;
11 }
12}
13
14let count: number = 0; // Initialize a counter for the nodes that meet the condition.
15
16function dfs(node: TreeNode | null): number {
17 if (!node) {
18 return 0; // If the current node is null, return 0 as it contributes nothing to the sum.
19 }
20
21 // Recursively compute the sum of descendants for the left and right subtrees.
22 const leftSum = dfs(node.left);
23 const rightSum = dfs(node.right);
24
25 // If the node's value equals the sum of its descendants, increment the count.
26 if (node.val === leftSum + rightSum) count += 1;
27
28 // Return the sum of the node's value and its descendants' values.
29 return node.val + leftSum + rightSum;
30}
31
32// This function counts the number of nodes in a binary tree
33// where the node value is equal to the sum of the values of its descendants.
34function equalToDescendants(root: TreeNode | null): number {
35 count = 0; // Reset count to 0 for each call to account for different binary tree inputs.
36
37 // Start the Depth-First Search (DFS) traversal from the root and count the nodes satisfying the condition.
38 dfs(root);
39
40 // Return the final count of nodes.
41 return count;
42}
43
Time and Space Complexity
The given Python code defines a method equalToDescendants
which checks each node in a binary tree to see if its value is equal to the sum of the values of its descendant nodes. Let's analyze the time and space complexity of this code:
Time Complexity:
The time complexity of the equalToDescendants
method is O(n)
, where n
is the number of nodes in the binary tree. This efficiency results from the fact that the method involves a depth-first search (DFS), visiting each node exactly once. The comparison and addition operations at each node occur in constant time, so the overall time complexity is linear concerning the number of nodes.
Space Complexity:
The space complexity of the code is O(h)
, where h
is the height of the tree. This space is used by the call stack during the execution of the DFS. In the worst case, when the binary tree degrades to a linked list (completely unbalanced), the space complexity will be O(n)
. In the best case, with a completely balanced tree, the space complexity will be O(log n)
because the height of the tree would be logarithmic relative to the number of nodes.
Therefore, the space complexity ranges from O(log n)
to O(n)
depending on the shape of the tree.
Learn more about how to find time and space complexity quickly using problem constraints.
Consider the classic dynamic programming of longest increasing subsequence:
Find the length of the longest subsequence of a given sequence such that all elements of the subsequence are sorted in increasing order.
For example, the length of LIS for [50, 3, 10, 7, 40, 80]
is 4
and LIS is
[3, 7, 40, 80]
.
What is the recurrence relation?
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
Binary Search Tree Intro Definition A Binary Search Tree or BST is a rooted binary tree with the value of each internal node being greater than all the values in the respective node's left subtree and less than the ones in its right subtree An empty tree is a BST since it satisfies the
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.