Given a binary tree where each node has a value and each node's value is either equal to the value of its parent or strictly greater than the value of its parent, find the second smallest value in the tree. The smallest value will be the root node value. If no second smallest value exists, return -1.
For example, consider the following binary tree:
2
/ \
2 5
/ \ / \
5 7 5 7
In this case, the smallest value is 2. The next smallest value is 5. Therefore, the function should return 5.
Consider another example:
2
/ \
2 2
/ \ / \
2 2 2 2
Here, all node values are 2. There is no second smallest value. The function should return -1.
Constraints:
[1, 1000]
.0 <= Node.val <= 2^31 - 1
root.val
is always the smallest value in the tree.Given a binary tree where each node has a value, and each node's value is either equal to the value of its parent or strictly greater than the value of its parent, find the second smallest value in the tree. The smallest value will be the root node value. If no second smallest value exists, return -1.
For example:
2
/ \
2 5
/ \ / \
5 7 5 7
In this case, the smallest value is 2. The next smallest value is 5. Therefore, the function should return 5.
Consider another example:
2
/ \
2 2
/ \ / \
2 2 2 2
Here, all node values are 2. There is no second smallest value. The function should return -1.
A straightforward approach is to traverse the entire tree, collect all unique values into a set, remove the smallest value (which is the root's value), and then return the minimum value from the remaining set, if the set is not empty. If the set is empty, it means there is no second smallest value, and we return -1.
class TreeNode:
def __init__(self, val=0, left=None, right=None):
self.val = val
self.left = left
self.right = right
def find_second_minimum_value_naive(root: TreeNode) -> int:
unique_values = set()
def traverse(node):
if node:
unique_values.add(node.val)
traverse(node.left)
traverse(node.right)
traverse(root)
unique_values.discard(root.val)
if not unique_values:
return -1
else:
return min(unique_values)
Since we know that the root node contains the smallest value, we can optimize the search. We only need to find the smallest value that is greater than the root's value. We can perform a depth-first search (DFS) and stop traversing down a path as soon as we find a value greater than the root's value. If we encounter a node with a value equal to the root's value, we continue traversing down that path.
class TreeNode:
def __init__(self, val=0, left=None, right=None):
self.val = val
self.left = left
self.right = right
def find_second_minimum_value(root: TreeNode) -> int:
smallest = root.val
second_smallest = float('inf')
def traverse(node):
nonlocal second_smallest
if not node:
return
if smallest < node.val < second_smallest:
second_smallest = node.val
elif node.val == smallest:
traverse(node.left)
traverse(node.right)
traverse(root)
return second_smallest if second_smallest != float('inf') else -1
The optimal solution gracefully handles the mentioned edge cases. If the tree contains only the smallest value, second_smallest
remains float('inf')
, and the function returns -1.