Create 3 different algorithms for making change (MC)
The main problem for this project is making change for a given amount with the fewest possible coins, using a given set of possible coin values. Example inputs and outputs:
Example 1:
input1: target = 10, coins_val = [5, 1]
output1: 2 (2 nickels make up 10)
Example 2:
input2: target = 35, coins_val = [10, 5, 1]
output2: 4 (3 dimes and 1 nickel)
The coin values can be passed a non-coin value (e.g., 4, 17), and the program should still return the optimal number of coins that can be used to make change for target amount.
The greedy algorithm takes the best option possible at that moment and returns a solution. A simple greedy algorithm of this MC problem will:
- Get the largest coin that is smaller than the current target
- Keep updating target and keep count until target reaches 0
Pseudocode:
count = 0
while target > 0
c = larget coin value < current target
count ++
target -= c
To find the largest coin value within the range of the current target, we will need another loop which iterates through the coin_val array.
The recursive algorithm for MC uses a table to store and reuse returned values (memoization). To be more specific here is a demonstration of a non-memoized purely recursive algorithm:
input: target=8, coins_val=[5, 4, 1]
output: 2 (2 fours)
Note that in Fig.1, the node 4 in the left bottom would have more nodes, if not for the limit in how many items I could add in free version of LucidChart. Here we can see that a lot of the same nodes exist, for example 3. For a subtarget of 3, a coin of value 1 can be used three times. This same recursive call is being done three different times, as demonstrated in Fig.1. Fig.2 shows a recursive tree of the memoized version:
Fig.2: Recursing tree for memoized approachIn Fig.2, we can see that there is less recursive call than Fig.1. This is because the optimal solution for the subtargets is being memoized (stored in a table), so that when the same value is encountered later in its execution it can refer back to the table. Here is a brief pseudocode:
func divideAndConquer(target, table, coins_val)
if target == 0
return 0
else if table contains target
return table[target]
min = infinity
for i from 0 to coins_val.length
sol = divideAndConquer(target-i, table, coins_val, c+1)
if (table[target] does not contain value) or (sol < min)
min = sol + 1
table[target] = sol
return min
Dynamic Programming is also known as the bottom-up approach, and in MC, we are going to build a table containing the optimal solutions of subproblems.
input: target=8, coins_val=[5, 4, 1]
table:
i count
1 1
2 2
3 3
4 1
5 1
6 2
7 3
8 2
output: 2 (2 fours)
In this is example, it first calculates the optimal solution if the target was 1. Then it uses this value, to calculate the optimal solution of target 2. However, when the optimal solution is smaller than the one derived from previous solution, it updates the table value to the optimal solution. By the time it returns the solution for the actual target, the table would contain an optimal solution for every subtarget value smaller than the actual target value. Here is the pseudocode for this algorithm:
table.length = target
for i from 0 to table.length
for j in coin_vals
if j < i
break
else if j == coin_vals
table[i] = 1
else
if (table[i] has no sol) or (1 + table[i-j] < ]table[i])
table[i] = 1 + table[i-j]
return table[target]
int greedy(int val, int* coins, int n) {
int score = 0;
int i = 0;
while (val > 0) {
// making sure i never gets out of index
while (i < n-1 && coins[i] > val) {
i++;
}
val -= coins[i];
score += 1;
}
// score is returned only if exact change can be made
if (val == 0) {
return score;
}
return -1;
}
The greedy algorithm turned out to work almost flawlessly based on the pseudocode above. In the while loop inside the outer loop, I am checking for every coin value that is smaller than the target. If the coin value is bigger than the target, then that coin cannot be used in the first place, hence we i++
, and do nothing else. In my previous version of this algorothm, I had my inner while loop as below:
// making sure i never gets out of index
while (coins[i] > val) {
i++;
}
This was working almost flawlessly until I found a case where it did not work. The reason this was happening was because whenever coins[last_index] > val
was called, last_index + 1
is out of range. To solve this problem, I added one condition, i < n-1
in the while loop to prevent from indexing out of range. This solved the error, and int greedy
now works well.
int divide_and_conquer(int val, int* coins, int n, int* table) {
if (val == 0) {
return 0;
}
// memoization
if (table[val] != -1) {
return table[val];
}
int min = -1;
for (int i = 0; i < n; i++) {
if (coins[i] > val) {
continue; // if coin value is bigger continue to the next coin value in array
}
int r = divide_and_conquer(val-coins[i], coins, n, table);
// update m if there is not yet a sol or this is a better sol
// check if the last recursive call returned a sol
if (r != -1) {
if (min == -1 || (min != -1 && r < min)) {
min = r+1;
}
}
}
table[val] = min;
return min;
}
The two base cases are:
- when the same amount of sum of coin values has been reduced from the target
- when the table for the target value already contains a solution
Whenever it fits in neither of the base cases, it finds a solution for the smaller targets and stores the value in the table. However, when adding a value at table[i]
, where i is the target value at that recursive call, it should run the following check:
if (r != -1) {
if (min == -1 || (min != -1 && r < min)) { //do }}
Before any of the algorithms are ran, the table is initialized in the int main
function, with table.length = target
and table[each index] = -1
. When a table at index i contains a -1, this means that there has been no solution found yet, for a target value of i. In the first condition if (r != -1)
, checks whether if the last recursive call found a solution or not. If the last recursive call did not find a solution, then neither will the successive recursive calls. However, if the previous recursive call returned a solution, then we move on to the next condition if (min == -1 || (min != -1 && r < min))
. This checks if the current value does not have a solution, or if it has one but the current solution is a better one than the previously obtained. If both of these conditions are true, then we update the solution and move on the next recursive call.
int dyn_prog(int val, int* coins, int n, int* table, int* sol) {
for (int i = 1; i < val+1; i++) {
// looking through coins from smallest
for (int j = n-1; j >= 0; j--) {
// break if coin is bigger than the value
if (i == coins[j]) {
table[i] = 1;
sol[i] = coins[j];
break;
} else if (coins[j] > i) {
break;
}
// update table if no sol is found or
if (table[i] == -1 || (table[i] != -1 && (1 + table[i-coins[j]] < table[i]))) {
if (table[i-coins[j]] == -1) {
continue;
}
table[i] = 1 + table[i-coins[j]];
}
}
}
return table[val];
}
Similar to the divide and conquer solution, in this dynamic programming solution, we are also going to use the same table initialized in the same way. Some implementation decisions I made:
- Looking through the coins from smallest to largest
The input array coins_val contains the values in reverse-sorted order (from biggest to smallest). If I iterated through the array normally from j=0 to j=len(coins), there is higher chance that we will be iterating through coin values that are bigger than the current target. Although skipping to the next iteration does not take a lot of time, it felt more intuitive for me to iterate through the coins array from j=len(coins) to j=0, because in this way we don't even have to go through unnecessary coin values.
- Finding the optimal solution for each target
Same idea with recursive approach, where we are building our answers based on previous answers. The implementation is very similar, as we are checking if the
table[i-coins[j]]
, which is the previous subproblem, contains a solution. If it does then,table[i]
should not update it's solution based ontable[i-coins[j]]
unless it has it's own independent solution. Furthermore, iftable[i]
does not already have a solution, or if it finds a better solution on the way, then the program would update the value attable[i]
.
Time Complexity: O(n)
Space Complexity: O(1)
One important thing to note about this greedy algorithm for MC, is that it doesn't necessarily give us an optimal solution. Here is an example:
input: target = 8, coins_val=[5, 4, 1]
demonstration:
1. new_target = 8-5 = 3, coin_count = 1
2. new_target = 3-1 = 2, coin_count = 2
3. new_target = 2-1 = 1, coin_count = 3
3. new_target = 1-1 = 0, coin_count = 4
output: 4, sol=[5, 1, 1, 1]
The output of the greedy algorithm, when processing the input above is a 4, however the optimal output in this case is 2, where sol=[4, 4]
. This algorithm takes whatever the biggest coin value it can fit in the current target, regardless of whether it can have a better solution or not. This is the reason why for some cases, this doesn't give us the optimal solution.
The time complexity for this algorithm is O(n*k), where n is the target value, and k is the length of the coins_val array. In this case, k is always going to be a finite number, so this can be treated as a constant. In result, the time complexity becomes O(kn), where k is a coefficient on n, which reduces to O(n). Furthermore the space complexity is O(1), because we are not storing anything except for the 2 variables score
and i
.
Time Complexity: O(n)
Space Complexity: O(n)
Without memoization, the divide and conquer method will take too long, and at some point it ends being stack overflowed. For example for an input such as target=3588, coins_val=[100, 25, 5, 1]
, the non-memoized approach would not work.
With memoization, the time complexity of int divide_and_conquer
becomes O(n). To be more specific it is O(k(n/j)), where k is the length of coins_val, n is the target value, and j is the value of the coin being used. The recursive call divide_and_conquer(val-coins[i], coins, n, table)
runs n/coins[i] times in general, however at most n times, considering that the smallest possible coin value is 1. The outer loop for (int i = 0; i < n; i++)
loops through the coins_val, which means it runs k times. As mentioned previously, k is conventionally a finite number which can be treated as a constant. Hence, the time complexity reduces to O(n), which is the same as the greedy algorithm. Space complexity on the other hand is worse than the greedy algorithm. In memoized divide and conquer method we are keeping the solutions for every subtargets in a table. This table does not necessarily have a length of n, however worst case, the length is n. Thus the space complexity is O(n).
Time Complexity: O(n)
Space Complexity: O(n)
Dynamic programming approach, similar to divide and conquer approach, always gives out a optimal solution. For example:
input: target = 8, coins_val=[5, 4, 1]
table:
i val
1 1
2 2
3 3
4 1
5 1
6 2
7 3
8 2
output: 2, sol=[4, 4]
All the subtargets leading to the final target is going to contain an optimal solution, and so it is guaranteed to have an optimal solution once it reaches the final target. The time complexity of this algorithm is again, O(n) for similar reason with the other two algorithm. It uses a nested for loop, however the inner loop only runs a finite time, hence O(n). The space complexity of this algorithm is O(n) because it requires a table of length n, and another data structure named sol
, which also has a length of n. sol
is used to build the solution containing all the used coin values. With both of these data stored, the precise space complexity is O(2n), which is the same as O(n).