Solution: House Robber III
Let's solve the House Robber III problem using the Backtracking pattern.
Statement
A thief has discovered a new neighborhood to target, where the houses can be represented as nodes in a binary tree. The money in the house is the data of the respective node. The thief can enter the neighborhood from a house represented as root
of the binary tree. Each house has only one parent house. The thief knows that if he robs two houses that are directly connected, the police will be notified. The thief wants to know the maximum amount of money he can steal from the houses without getting caught by the police. The thief needs your help determining the maximum amount of money he can rob without alerting the police.
Constraints:
- The number of nodes in the tree is in the range .
-
node.data
Solution
So far, you’ve probably brainstormed some approaches and have an idea of how to solve this problem. Let’s explore some of these approaches and figure out which one to follow, based on considerations such as time complexity and any implementation constraints.
Naive approach
A naive approach to solving this problem involves calculating the sum of every possible valid combination of houses that can be robbed. A valid combination is one in which the thief does not rob two houses that are directly connected. After calculating all possible valid combinations, we can return the sum of the combination of houses that produces the maximum amount of money. This approach requires evaluating all possible ways to rob the neighborhood, which results in exponential time complexity. We can improve the algorithm’s efficiency by using backtracking.
Optimized solution using backtracking
An optimized approach to solve this problem is using backtracking, which involves recursively exploring all possible combinations and backtracking whenever we encounter an invalid combination. By doing so, we can avoid evaluating redundant solutions and reach the optimal combination more efficiently.
Our solution aims to find the maximum amount of money a thief can rob from the houses in a binary tree without alerting the police. The rob
function takes the root
node of the binary tree as input, calls the heist
function, and returns the maximum amount of money that the thief can rob from the binary tree.
The heist
function recursively calculates the maximum amount of money that can be robbed from a subtree rooted at a node. It returns a pair containing the values of includeRoot
and excludeRoot
.
includeRoot
contains the maximum amount of money that can be robbed from the subtree rooted at the node with theroot
node included.excludeRoot
contains the maximum amount of money that can be robbed from the subtree rooted at the node with theroot
node excluded.
The function proceeds through the following steps:
-
If the tree is empty, the heist function returns
[0, 0]
, which represents theincludeRoot
andexcludeRoot
, respectively. -
Otherwise, it recursively calculates the maximum amount of money that can be robbed from the left and right subtrees of the
root
node. -
Then, it stores the maximum amount of money that can be robbed from the
root
node in theincludeRoot
. It computes the maximum amount by adding the value of theroot
node to the following:- The maximum amount of money that can be robbed from the left subtree, excluding their respective parent node.
- The maximum amount of money that can be robbed from the right subtree, excluding their respective parent node.
-
Alternatively, it computes the sum of the maximum amount of money that can be robbed from both the left subtree and right subtree of the
root
node, excluding theroot
node itself, and store it in theexcludeRoot
. -
Finally, the
heist
function returns a pair containing the values ofincludeRoot
andexcludeRoot
.
The rob
function then returns the maximum value from the pair obtained by the heist
function, which is the maximum amount of money the thief can rob from the houses in a binary tree without alerting the police.
Let’s look at the following illustration to get a better understanding of the solution:
Let’s implement the algorithm as discussed above:
public class Main {public static int[] heist(TreeNode<Integer> root) {if (root == null) {return new int[]{0,0};}int[] leftSubtree = heist(root.left);int[] rightSubtree = heist(root.right);int includeRoot = root.data + leftSubtree[1] + rightSubtree[1];int excludeRoot = Math.max(leftSubtree[0], leftSubtree[1]) + Math.max(rightSubtree[0], rightSubtree[1]);return new int[] {includeRoot, excludeRoot};}public static int rob(TreeNode<Integer> root) {int[] result = heist(root);return Math.max(result[0], result[1]);}public static void main(String[] args) {List<List<TreeNode<Integer>>> listOfTrees = Arrays.asList(Arrays.asList( new TreeNode<Integer>(10), new TreeNode<Integer>(9), new TreeNode<Integer>(20), new TreeNode<Integer>(15), new TreeNode<Integer>(7)),Arrays.asList( new TreeNode<Integer>(7), new TreeNode<Integer>(9), new TreeNode<Integer>(10), new TreeNode<Integer>(15), new TreeNode<Integer>(20)),Arrays.asList( new TreeNode<Integer>(8), new TreeNode<Integer>(2), new TreeNode<Integer>(17), new TreeNode<Integer>(1), new TreeNode<Integer>(4), new TreeNode<Integer>(19), new TreeNode<Integer>(5)),Arrays.asList( new TreeNode<Integer>(7), new TreeNode<Integer>(3), new TreeNode<Integer>(4), new TreeNode<Integer>(1), new TreeNode<Integer>(3)),Arrays.asList( new TreeNode<Integer>(9), new TreeNode<Integer>(5), new TreeNode<Integer>(7), new TreeNode<Integer>(1), new TreeNode<Integer>(3)),Arrays.asList( new TreeNode<Integer>(9), new TreeNode<Integer>(7), null, null, new TreeNode<Integer>(1), new TreeNode<Integer>(8), new TreeNode<Integer>(10), null, new TreeNode<Integer>(12)));List<BinaryTree<Integer>> inputTrees = new ArrayList<>();for (List<TreeNode<Integer>> ListOfNodes : listOfTrees) {BinaryTree<Integer> tree = new BinaryTree<>(ListOfNodes);inputTrees.add(tree);}int x = 1;for (BinaryTree<Integer> tree : inputTrees) {System.out.println(x + ".\tInput Tree:");Print.displayTree(tree.root);x++;System.out.println("\n\tMaximum amount we can rob without getting caught: " + rob(tree.root));System.out.println(new String(new char[100]).replace('\0', '-'));}}}
Solution summary
-
If the tree is empty, return .
-
Otherwise, recursively calculate the maximum amount of money that can be robbed from the left and right subtrees of the
root
node. -
Compute the maximum amount of money that can be robbed with the parent node included.
-
Compute the maximum amount of money that can be robbed with the parent node excluded.
-
Return the maximum value from the following:
- The maximum amount of money that can be robbed with the parent node included.
- The maximum amount of money that can be robbed with the parent node excluded.
Time complexity
The time complexity of this solution is , where is the number of nodes in the tree, since we visit all nodes once.
Space complexity
The space complexity of this solution is , since the maximum depth of the recursive call tree is the height of the tree, which is in the worst case, and each call stores its data on the stack.
Level up your interview prep. Join Educative to access 70+ hands-on prep courses.