Lesson 7
Mastering Interview Problems with Binary Search Trees
Introduction to the Lesson

Welcome to today's lesson! In this practice-based session, we'll unravel the mysteries of Binary Search Trees (BSTs). We will focus on two well-known interview problems related to BSTs. Solving these problems will solidify your understanding of trees and provide you with hands-on experience that will be invaluable during future job interviews. Think of this as mock interviews before the real one! So, let's put on our problem-solving hats and dive right in!

Problem 1: Checking the Balance of a Binary Search Tree

Here is the first problem: we need to write a function that checks if a BST is balanced. The tree is balanced if for each vertex, the size of the left subtree differs from the size of the right subtree by at most 1.

Problem 1: Problem Actualization

You might wonder, "Why do we need to do this?". Well, this problem frequently appears in job interviews because checking the balance of a tree optimizes search operations.

Interestingly, consider trying to balance on one foot. If the weight on your left and right sides is not equal, you'll topple over, right? Similarly, a binary tree is considered balanced when the left and right sides are equal, or at least the difference is no more than one. Isn't that fascinating?

Problem 1: The Naive Approach

So, let's discuss the naive approach to solving this problem. One could start by calculating the heights of all subtrees and then checking whether the heights of each node's left and right subtrees differ by no more than one. While this approach is functional, it is inefficient since it involves traversing the entire tree multiple times and making duplicate calculations.

Problem 1: Efficient Approach Explanation

But don't worry, there's a more efficient way! Instead of traversing the tree multiple times, we can use recursion to do it all in one sweep. We calculate the heights of the subtrees while simultaneously checking the balance condition. So, essentially, we're hitting two birds with one stone!

Problem 1: Solution Building

Let's dive into the actual code. Checking if a tree is balanced involves examining all its nodes. However, we can approach this more efficiently. Instead of calculating the height of each node separately, which involves repeated traversal, we can implement a check_balance function that computes both the height of the tree and checks balance in one recursive traversal. The function should return an indicator is_balanced = False when the tree is unbalanced to allow for early termination without visiting all nodes. Here's what the updated solution would look like:

Python
1def is_balanced(root) -> bool: 2 # returns (height, is_balanced) 3 def check_balance(node) -> (int, bool): 4 if node is None: 5 return 0, True 6 7 left_height, left_balanced = check_balance(node.left) 8 if not left_balanced: 9 return -1, False 10 11 right_height, right_balanced = check_balance(node.right) 12 if not right_balanced: 13 return -1, False 14 15 height = max(left_height, right_height) + 1 16 is_balanced = abs(left_height - right_height) <= 1 17 return height, is_balanced 18 19 height, balanced = check_balance(root) 20 return balanced

This solution performs with O(n)O(n) time complexity, where nn is the number of nodes in the tree, as each node is visited only once.

Problem 2: Identify the K-th Smallest Element in a Binary Search Tree

Imagine needing to identify the player with the k-th best result while constructing a leaderboard for a game. We're expected to find this kth smallest element in a Binary Search Tree (BST) where the players' scores are stored for efficient retrieval.

Problem 2: Actualization

We're dealing with a BST where each player's score represents a node. Our goal is to identify the k-th smallest score, which translates to the k-th smallest element within the BST.

Problem 2: Naive Approach

A simplistic, blunt approach involves storing all the elements in an array. We then sort it and return the kth element. This brute force method, however, has a time complexity of O(nlogn)O(n \log n) due to the sorting operation. It also necessitates extra space, thus revealing a space complexity of O(n)O(n). There must be a more efficient method, right?

Problem 2: Efficient Approach

A more efficient and memory-friendly strategy to address this task entails employing a recursive method without explicit in-order traversal.

This approach involves counting the number of nodes in the left subtree of the root. If the count of nodes matches k - 1, the root value is the k-th smallest element we're looking for. If the k is smaller or equal to the count, the k-th smallest is in the left subtree. If the k is larger, then the k-th smallest is in the right subtree, and we adjust k accordingly.

Problem 2: Solution Implementation

Let's create the function kthSmallest, where we utilize a recursive solution to retrieve the k-th smallest score directly:

Python
1def kthSmallest(root, k): 2 # The number of nodes in the left subtree of the root 3 left_nodes = countNodes(root.left) if root else 0 4 5 # If k is equal to the number of nodes in the left subtree plus 1, 6 # That means we must return the root's value as we've reached the k-th smallest 7 if k == left_nodes + 1: 8 return root.val 9 # If there are more than k nodes in the left subtree, 10 # The k-th smallest must be in the left subtree. 11 elif k <= left_nodes: 12 return kthSmallest(root.left, k) 13 # If there are less than k nodes in the left subtree, 14 # The k-th smallest must be in the right subtree. 15 else: 16 return kthSmallest(root.right, k - 1 - left_nodes) 17 18def countNodes(root): 19 if not root: 20 return 0 21 return 1 + countNodes(root.left) + countNodes(root.right)

Please note that this code snippet relies on a helper function, countNodes, which returns the count of nodes in a given tree. The main function compares the total nodes in the left subtree with k and decides whether the kth smallest is in the left subtree or right subtree or if it is the root itself.

This approach has O(k)O(k) time complexity, as we visit at most k vertices in the tree.

Lesson Summary

Well done! You've navigated the ins and outs of binary search trees! You've successfully learned how to validate the balance of a BST and how to find the second minimum value in a BST. These concepts are highly relevant for any software engineering job interview. Let's give you a round of applause for mastering these concepts!

Practice Exercises

Our immersive journey into binary search trees does not end here. Up next, we have some practice exercises based on these problems for you to solve. These are designed to reinforce your understanding and bolster your confidence in handling real interview questions about BSTs. So, brace yourself, and let's dive right into the exercises. Remember, practice makes perfect. Happy coding!

Enjoy this lesson? Now it's time to practice with Cosmo!
Practice is how you turn knowledge into actual skills.