You are a professional robber planning to rob houses along a street. Each house has a certain amount of money stashed. All houses at this place are arranged in a circle. That means the first house is the neighbor of the last one. Meanwhile, adjacent houses have a security system connected, and it will automatically contact the police if two adjacent houses were broken into on the same night.
Given an integer array
Output: 3
Explanation: You cannot rob house 1 (money = 2) and house 3 (money = 2) as these two are adjascent.
So, the total amount will be 3.
Output: 4
Explanation: You can rob house 1 (money = 1), house 3 (money = 3).
So, the total amount will be 4
Output: 3
Explanation: You can rob house 3 (money = 3)
So, the total amount will be 3
Contents
Solution intution from previous house robber problem Solution 1 - using brute force approach Solution 2 - using DP memoization Solution 3 - using DP Bottom-Up approach using tabulation Solution 4 - using DP Bottom-Up approach memory optimized
This problem is a similar to the previous
-
First time using houses from
1,......n-1 , i.e. excluding last house, this way if you consider 1st house then you are ignoring last house since they are adjascent to each other. -
Second time using houses from
2,.....n , i.e. excluding first house, this way you are considering last indexed house, but ignoring the first house since they are adjascent to each other.
Then, finally take the maximum from these two results.
In this approach, we will apply the solution intution discussed above to the brute force approach, that is to
compute all possible combinations with conditions given in the problem, we will run this process on sub arrays
-
We will create a function
bruteForceApproach that takesnums array as input argument. -
Create another helper function
bruteForceApproachRec that takenums andstart variable to indicate the start index of the sub array we should be considering andend variable to indicate the end index of sub array we should be considering.- Call this function recursively, by moving start to +2 index considering you want to rob current house and +1 index considering you dont want to rob the current house, and take the maximum out of these two possibilities.
- Base case will be, when it reaches end of array or goes out of bounds as we are trying with +1 and +2 index from every start index, and when start index becomes greater than end index, then we simply return 0.
-
Call
bruteForceApproachRec two times. First, withstart =0, end = nums.length-2 , i.e. ignoring last indexed house since we are considering first house. Second, withstart =1, end = nums.length--1 , i.e. ignoring first indexed house since we are considering last house.
Complexity Analysis:
Time complexity: Above code runs in O(2n) time where
Space complexity: O(n) as we are making recursive loops
From the above brute force approach, we can notice that there are some overlapping problems.
-
So, in this approach, we will be caching sub problem result into a
HashMap withstart index as key and its result as value. When a sub problem is repeated and found in cache, we will return it, otherwise we will call the function recursively and cache its result.
Complexity Analysis:
Time complexity: Above code runs in O(n) time where
Space complexity: O(n) for the cache we are maintaining and stack for recursive calls.
In this approach we will look at solving the problem using Dynamic programming using tabulation caching mechanism.
The time and memory complexity will be same as above solution using memoization,
but we can notice a pattern to improve memory complexity to constant in later solution.
Intution behind this solution is, when we are at
We will repeat this solution for house indexed
Base cases for this solution will be
-
dp[0] = nums[0] , because at 0th index, maximum amount you can rob is just that house. -
dp[1] = Max(nums[0], nums[1]) , because at 0th indexed house has higher amount, you can rob that house and skip index 1 house or rob index 1 house and skip index 0 house.
Finally, take the last element from
Since we need to call this for houses indexed
Complexity Analysis:
Time complexity: Above code runs in O(n) time where
Space complexity: O(n) since we have a
If we look at above DP solution using tabulation, when calculating maximum amount you can rob at index
-
In this approach, we will create two variables
m1 andm2 to hold previous values for i-2nd step and i-1st indexed houses. -
We will initialize
m1 = 0 andm1 = 0 . -
Next, for each index
i compute the maximum that can be robbed usingMax(nums[i]+m1, m2) , store this into a temporary variable, then make this value as newm2 and current m2 tom1 for next iteration.
Since we need to call this for houses indexed
Complexity Analysis:
Time complexity: Above code runs in O(n) time where
Space complexity: O(1) since we are only using m1, m2 and a temp variables.
Above implementations source code can be found at