2792. Count Nodes That Are Great Enough
Problem Description
In this problem, we are given a binary tree where each node has an integer value. Alongside, we are given an integer k
. The task is to find and count the nodes in the binary tree that are considered "great enough". A node is "great enough" if it meets the following criteria:
- The subtree rooted with this node has at least
k
nodes. - The value of this node is greater than the value of at least
k
nodes in its subtree. We must explore the tree and count how many nodes satisfy these conditions.
Intuition
The intuition behind the solution is to perform a depth-first search (DFS) starting from the root node, to explore the entire tree. While exploring, we need to keep track of the nodes and their values in a way that enables us to check if a node is "great enough" according to the given conditions.
To check if a node is "great enough", we accumulate the greatest values we encounter in its subtree, including the node itself. We use a min-heap data structure, which allows us to efficiently keep track of the largest elements (since in Python's heapq
the smallest elements come first, we push negatives of the values so that when we pop, we pop the smallest of the largest values).
Here's the key to our approach:
- Perform DFS from the root.
- For each node, merge the heaps of its left and right children to congregate their greatest values.
- Ensure our heap does not exceed
k
elements. - Once merged, we compare the negative of the top of our min-heap (which is the
k
th largest value) with current node's value to determine if our node is "great enough". - If it is, increment our counter
ans
. - Finally, push the negative of the current node's value onto our heap (to include the current node in the subtree for future parent nodes) before returning the heap to its parent call.
The main advantages of this approach are:
- Space efficiency, as we carry a single heap up the tree while we are gathering subsets of values.
- Time efficiency, since heaps provide O(log k) pushing and popping, making the overall complexity better compared to sorting the values or other brute force methods that might seem tempting at first glance.
Solution Approach
The provided solution uses a Depth-First Search (DFS) algorithm in combination with a min-heap to solve the problem. Here is a step-by-step explanation of the implementation:
-
A DFS
dfs
function is defined, which will be called recursively to explore the tree nodes. At each node, this function will return a min-heap containing the largest values (negated) from its subtree, with size at mostk
. -
Within this
dfs
function, when the base case is encountered (i.e., when aNone
node is encountered), an empty heap is returned, indicating that a leaf has been reached. -
When at a non-leaf node, DFS is called on both the
left
andright
children, and their returned heaps are merged. It's important to note that we're using thepush
helper function to maintain the heap's size at no more thank
. -
The
push
helper function works as follows: It pushes an element onto the heap and if the heap size is greater thank
after the push, it pops the smallest element (which is represented as the largest negative value). This ensures that only thek
largest (negated) elements remain. -
Back in the
dfs
function, after merging the heaps from the left and right subtrees, we check if the current node's value is "great enough". This check is done by ensuring that the heap size is exactlyk
and the top element of the heap (i.e., the smallest of thek
largest negated values) is less than the negated value of the current node. If these conditions are met, it means the current node is greater than at leastk
nodes in its subtree, and we increment our answerans
. -
Finally, the current node's value (negated) is pushed onto the heap, and the heap is returned to be merged in the parent node's DFS call.
-
After the DFS traversal completes, the counter
ans
holds the number of "great enough" nodes, and this value is returned as the solution to the problem.
Here is the pseudocode for the DFS and push function:
1def dfs(node):
2 if node is None:
3 return new empty heap
4 left_heap = dfs(node.left)
5 right_heap = dfs(node.right)
6 # Merge heaps
7 for value in right_heap:
8 push(left_heap, value)
9 # Check if the current node is "great enough"
10 if heap size is k and (-top of heap < node's value):
11 ans += 1
12 push(left_heap, -node's value)
13 return left_heap
14
15def push(heap, value):
16 add value to heap
17 if size of heap > k:
18 remove the smallest element from heap
Notice that the dfs
function's merge step iterates over the right heap and pushes its elements into the left heap. This merging avoids creating a new heap at each node, which saves memory and potentially decreases runtime due to fewer allocations.
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 use a small binary tree example and k = 2
to illustrate the solution approach.
Imagine a binary tree like this:
1 5 2 / \ 3 3 8 4 / \ \ 5 1 4 10
In this tree, we want to count the number of "great enough" nodes, i.e., nodes with at least k
nodes in their subtree and a value greater than at least k
nodes in their subtree.
-
Begin DFS at the root node (5). Since it's not
None
, we proceed. -
Perform DFS on the left child (3).
- Left child (3) is not
None
, we perform DFS on its children.- Left child (1) has no children (both
None
), so it returns an empty heap. - Right child (4) has no children (both
None
), so it returns an empty heap.
- Left child (1) has no children (both
- Merge returned heaps from 1 and 4 and push negated values:
left_heap
contains[-4, -3]
. - Now,
left_heap
has 2 elements (size isk
) andtop of heap
is -4, which is smaller than-3
. So, node 3 is "great enough". - Increment
ans
to 1 and push-3
into theleft_heap
before returning it.
- Left child (3) is not
-
Perform DFS on the right child (8).
- Right child (8) is not
None
, perform DFS on its left and right children.- It has no left child, so returns an empty heap.
- Right child (10) has no children (both
None
), so it returns an empty heap.
- Merge returned heaps and push negated values:
right_heap
contains[-10]
. - Heap size is less than
k
, so we cannot say node 8 is "great enough" yet. - Push
-8
into theright_heap
before returning it.
- Right child (8) is not
-
Back to the root now,
left_heap
contains[-4, -3]
andright_heap
contains[-10, -8]
. Merge them.- After merging, the combined heap will have
[-10, -8, -4, -3]
. We need to maintain only the largestk
values, so we pop off -3 to getleft_heap
as[-10, -8, -4]
. - The
top of heap
is now -4, which is smaller than-5
. Thus, the root node (5) is "great enough". - Increment
ans
to 2. Push-5
intoleft_heap
and returnleft_heap
.
- After merging, the combined heap will have
-
After DFS traversal, the
ans
is 2, representing the number of "great enough" nodes (nodes 3 and 5).
Thus, the output for our binary tree example with k = 2
is 2, meaning there are 2 nodes that are "great enough" by those criteria.
Solution Implementation
1# Definition for a binary tree node.
2class TreeNode:
3 def __init__(self, val=0, left=None, right=None):
4 self.val = val
5 self.left = left
6 self.right = right
7
8class Solution:
9 def countGreatEnoughNodes(self, root: Optional[TreeNode], k: int) -> int:
10 from heapq import heappush, heappop
11
12 # Helper method to maintain a heap of size 'k'
13 def push_to_heap(heap, val):
14 # Add 'val' to 'heap' and ensure the heap size does not exceed 'k'
15 heappush(heap, val)
16 # If the heap size is > k, pop the smallest element
17 if len(heap) > k:
18 heappop(heap)
19
20 # DFS traversal to count the great enough nodes
21 def dfs(node):
22 if node is None:
23 # If the node is None, return an empty heap
24 return []
25 # Recursively traverse left and right subtrees
26 left_heap = dfs(node.left)
27 right_heap = dfs(node.right)
28 # Combine heaps from children and maintain size 'k'
29 for val in right_heap:
30 push_to_heap(left_heap, val)
31 # Check if current node's value is greater than k-th value (if heap size is k)
32 if len(left_heap) == k and -left_heap[0] < node.val:
33 # If condition satisfies, increment the result
34 nonlocal count
35 count += 1
36 # Add current node's value to the heap (negated to convert to max-heap)
37 push_to_heap(left_heap, -node.val)
38 # Return the updated heap
39 return left_heap
40
41 # Initialize count of great enough nodes as 0
42 count = 0
43 # Start DFS from the root
44 dfs(root)
45 # Return the final count
46 return count
47
1class Solution {
2 private int greatEnoughCount; // Counter to keep track of nodes meeting the condition
3 private int threshold; // The value k used as a threshold to compare node values
4
5 /**
6 * Counts the number of nodes in a binary tree where the node's value
7 * is greater than the k-th largest value of all nodes from the root to that node.
8 *
9 * @param root Root of the binary tree.
10 * @param k The k-th largest value to be considered.
11 * @return The count of nodes meeting the mentioned criterion.
12 */
13 public int countGreatEnoughNodes(TreeNode root, int k) {
14 this.threshold = k;
15 this.greatEnoughCount = 0;
16 depthFirstSearch(root);
17 return greatEnoughCount;
18 }
19
20 /**
21 * Performs a depth-first search on the binary tree, examining each node
22 * and updating the greatEnoughCount when conditions are met using priority queue
23 * for finding the k-th largest value from the root to the current node.
24 *
25 * @param node The current node being examined.
26 * @return A priority queue containing the largest values encountered up to the current node,
27 * with size maintained at most k.
28 */
29 private PriorityQueue<Integer> depthFirstSearch(TreeNode node) {
30 // If we reach a null node, return a priority queue that orders elements in reverse order
31 if (node == null) {
32 return new PriorityQueue<>(Comparator.reverseOrder());
33 }
34
35 // Recursively traverse left and right subtrees
36 PriorityQueue<Integer> leftQueue = depthFirstSearch(node.left);
37 PriorityQueue<Integer> rightQueue = depthFirstSearch(node.right);
38
39 // Merge the right queue into the left queue
40 for (int value : rightQueue) {
41 leftQueue.offer(value);
42 // If the size of the queue exceeds k, remove the smallest element
43 if (leftQueue.size() > threshold) {
44 leftQueue.poll();
45 }
46 }
47
48 // If the k-th largest value is defined and is less than the current node's value
49 if (leftQueue.size() == threshold && leftQueue.peek() < node.val) {
50 greatEnoughCount++; // Increment our count
51 }
52
53 // Add the current node's value to the queue
54 leftQueue.offer(node.val);
55 // Ensure the size of the queue doesn't exceed k
56 if (leftQueue.size() > threshold) {
57 leftQueue.poll();
58 }
59
60 return leftQueue; // Return the priority queue holding the largest k values up to and including this node
61 }
62}
63
1#include <functional>
2#include <queue>
3
4// Definition for a binary tree node.
5struct TreeNode {
6 int val;
7 TreeNode *left;
8 TreeNode *right;
9 TreeNode() : val(0), left(nullptr), right(nullptr) {}
10 TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
11 TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
12};
13
14class Solution {
15public:
16 // This function returns the number of nodes in a binary tree 'root'
17 // whose values are greater than all of the last 'k' visited nodes
18 // in a depth-first search order.
19 int countGreatEnoughNodes(TreeNode* root, int k) {
20 int answer = 0;
21
22 // Define a lambda function that uses Depth-First Search (DFS) to iterate
23 // through the tree and keep track of the largest 'k' values found so far.
24 std::function<std::priority_queue<int>(TreeNode*)> dfs = [&](TreeNode* node) -> std::priority_queue<int> {
25 // If the node is null, return an empty priority queue.
26 if (!node) {
27 return std::priority_queue<int>();
28 }
29
30 // Recursively apply DFS to the left and right subtrees.
31 auto leftMaxes = dfs(node->left);
32 auto rightMaxes = dfs(node->right);
33
34 // Combine the max values from left and right subtrees.
35 while (!rightMaxes.empty()) {
36 leftMaxes.push(rightMaxes.top());
37 rightMaxes.pop();
38
39 // Maintain the size of the priority queue as 'k'.
40 if (leftMaxes.size() > k) {
41 leftMaxes.pop();
42 }
43 }
44
45 // Check if the current node's value is greater than the smallest
46 // value in the max 'k' values seen so far (means it's greater than
47 // all 'k' values because all others are larger).
48 if (leftMaxes.size() == k && leftMaxes.top() < node->val) {
49 answer++;
50 }
51
52 // Add the current node's value to the collection of max values.
53 leftMaxes.push(node->val);
54
55 // Ensure the size of the priority queue does not exceed 'k'.
56 if (leftMaxes.size() > k) {
57 leftMaxes.pop();
58 }
59
60 return leftMaxes;
61 };
62
63 // Start the DFS from the root of the tree.
64 dfs(root);
65
66 // Return the final count of nodes that meet the condition.
67 return answer;
68 }
69};
70
1// Definition for a binary tree node.
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
14// This function performs Depth-First Search on the binary tree to find 'k' largest values.
15const dfs = (node: TreeNode | null, k: number, answer: { count: number }): number[] => {
16 if (node === null) {
17 return [];
18 }
19
20 // Recursively traverse left and right children.
21 const leftMaxes = dfs(node.left, k, answer);
22 const rightMaxes = dfs(node.right, k, answer);
23
24 // Flatten left and right max values and sort them in descending order.
25 const maxes = leftMaxes.concat(rightMaxes).sort((a, b) => b - a).slice(0, k);
26
27 // If the current node's value is greater than all 'k' values seen so far, increment the count.
28 if (maxes.length === k && maxes[maxes.length - 1] < node.val) {
29 answer.count++;
30 }
31
32 // Add the current node's value to the list of maxes and return.
33 maxes.push(node.val);
34 return maxes.sort((a, b) => b - a).slice(0, k);
35};
36
37// This function returns the number of nodes in a binary tree 'root'
38// whose values are greater than all of the last 'k' visited nodes
39// in a depth-first search order.
40const countGreatEnoughNodes = (root: TreeNode | null, k: number): number => {
41 // Initialize count as a reference object to keep track during recursive calls.
42 let answer = { count: 0 };
43
44 // Start the DFS from the root of the tree.
45 dfs(root, k, answer);
46
47 // Return the final count of nodes that meet the condition.
48 return answer.count;
49};
50
51// Example usage:
52const rootNode = new TreeNode(2, new TreeNode(1), new TreeNode(3));
53const result = countGreatEnoughNodes(rootNode, 2);
54console.log(result); // Output should be based on the tree structure and 'k' value.
55
Time and Space Complexity
The time complexity of the given countGreatEnoughNodes
function is O(n * log(n))
, where n
is the number of nodes in the binary tree. The function performs a depth-first search (DFS) on the tree, visiting each node once (O(n)
). At each node, the push
function potentially performs a heap push and pop operation in the priority queue. The length of the priority queue is capped at k
, so each heap operation is O(log(k))
. Since k
can, at maximum, be as large as n
(the total number of nodes), in the worst-case scenario, this becomes O(log(n))
. When multiplied by the total number of nodes visited, we obtain O(n * log(n))
.
The space complexity of the function is O(n)
. This is because the recursive DFS function call stack could potentially store every node in a path from the root to a leaf in the case of a skewed binary tree. In addition, the priority queue l
also stores up to k
elements, but since k
is at most n
, it does not exceed O(n)
space. Thus, the overall space complexity is determined by the maximum between the height of the tree and the number length of the priority queue, which is O(n)
.
Is the following code DFS or BFS?
1void search(Node root) { 2 if (!root) return; 3 visit(root); 4 root.visited = true; 5 for (Node node in root.adjacent) { 6 if (!node.visited) { 7 search(node); 8 } 9 } 10}
Recommended Readings
Patterns The Shortest Path Algorithm for Coding Interviews The goal of AlgoMonster is to help you get a job in the shortest amount of time possible in a data driven way We compiled datasets of tech interview problems and broke them down by patterns This way we can determine the
Recursion Recursion is one of the most important concepts in computer science Simply speaking recursion is the process of a function calling itself Using a real life analogy imagine a scenario where you invite your friends to lunch https algomonster s3 us east 2 amazonaws com recursion jpg You first
Runtime Overview When learning about algorithms and data structures you'll frequently encounter the term time complexity This concept is fundamental in computer science and offers insights into how long an algorithm takes to complete given a certain input size What is Time Complexity Time complexity represents the amount of time
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.