Coin Change Problem - Dynamic Programming
Problem:
Given a certain amount of coins that associated values to them (ex. coins=[1,2,5]), determine the minimum amount of coins that we need to reach the fiven amount (ex. amount = 11).
NOTE: same coins can be chosen multiple times.
This problem can be solved using recursion, top-down DP, or bottom-up DP. We will go over top-down and bottom-up DP approaches.
Top-Down DP Approach
In the top-down DP approach, our root node would have the value 11. From there, we would make 1 decision with each possible coin and see what the result is. At that point we know that we made 1 decision and we chose 1 coin. Hence, we need to add 1 to our final amount of coins. We would repeate the same process until we reach value of 0 at some node. If a value in a node is to be negative, that node should not be created, since we do not have a way to reach teh given amoun using that combination of coins. We can see in the following image how that would look like. Here, the tree is not fully developed, but it is enough to get the point accross.
Figure 1 - top-down approach
In the red circles we can see the nodes that require the same computation. We solve this, of cource, using memoization.
Bottom-Up DP Approach
In the bottom-up DP approach, we compute answers to the smaller subproblems and build the higher level problems using the solutions to the smaller subproblems. In this particular problem, if we know the minimum amount of coins to reach the amount of 2, then we will need to make just one more decision to reach the amount of 4. Let’s see how it works.
- We will initialize our array L of size 11 and assign arbitrarily high values as place-holders. In this case that can be value 12.
- For amount 0, we do no need any coins, hence we can initialize L[0] = 0.
- In order to solve the problem of choosing the minimum amount of coins for the amount 11, we need to solve the same problem for the smaller amounts than 11 (subproblems), which will lead towards to solution for value of 11.
Figure 2 - bottom-up approach
The process of thinking is as follows: For index 1 –> we will try each coin from our list of possible coins, which will lead us to a new amount, old_amount - coni_value = new_amount. Try coin 1 => 1 - 1 = 0. Our subproblem turned to be the amount of 0, for which we know the answer = 0. Now, we know that we picked coin 1, which means that we made a decision to add that coin to the final amount of coins to reach amount 1. For amount of 0, we need 0 coins, we add the 1 coin of value 1 we chose, and our final amount of coins we need to reach the amount would be the minimum between previous value at index1 and our new solution => min(0+1, 12) = 1. Hence, our best solution for value 1 so far is to pick only 1 coin. Try coin 2 => 1 - 2 = -1 => we do not take this in consideration since we get negative amount for our subproblem. Try coin 5 => 1 - 5 = -4 => we do not take this one in consideration eaither.
Figure 3 - subproblem 2 (1 was with the amount of 0)
We move on onto the index 2, meaning the subproblem of amount 2. Repeat the process. Pick coint 1 => 2 - 1 = 1. Our new amount is 1. We have already computed the best amount of coins to reach the value of 1, which is 1. Since this value is 1 and we picked the coin 1 again, that is 1 + 1 = 2 coins picked to make the value of 2. The final test in the array = 8min(1+1, 12) = 2. Now, we pick coin with value 2 => 2 - 2 = 0. We, know that for the amount of 0, we need no coins, so we only picked one coin this time 0 + 1 = 1. Check with the previous best amount of coins for the value of 2 and choose the smaller one => min(0+1, 2) = 1. Hence, our new minimum amount of coins to reach teh value of w is 1 (we just picked the coin with value 2). Now we try coin 5 => 2 - 5 = -3. Not considering this one.
Figure 4 - subproblem 3
We move on onto the index 3, meaning the subproblem of amount 3. Repeat the process. Pick coint 1 => 3 - 1 = 2. Our new amount is 2. We have already computed the best amount of coins to reach the value of 2, which is 1. Since this value is 1 and we picked the coin 1 again, that is 1 + 1 = 2 coins picked to make the value of 2. The final test in the array = 8min(1+1, 12) = 2. Now, we pick coin with value 2 => 3 - 2 = 1. We, know that for the amount of 1, we only 1 coint to reach taht amount, so we picked two coins again 1 + 1 = 2. Check with the previous best amount of coins for the value of 2 and choose the smaller one => min(1+1, 2) = 2. The minimum amount of coins picked does not change. Now we try coin 5 => 3 - 5 = -2. Not considering this one.
Figure 5 - subproblem 4
Repeat the same process for the rest of the subproblems until we get the lowest amount of coins we need for the amount of 11.
Figure 6 - final solution
My Solution
Variation of this problem is when coins can be picked only once!
The Squid