WA5. Knapsack Problem - Dynamic Programming¶
Statement¶
A thief robbing a store can carry a maximum weight of W in their knapsack. There are n items and ith item weighs wi and is worth vi dollars. What items should the thief take to maximize the value of what is stolen?
The thief must adhere to the 0-1 binary rule which states that only whole items can be taken. The thief is not allowed to take a fraction of an item (such as ½ of a necklace or ¼ of a diamond ring). The thief must decide to either take or leave each item.
Input:
- Maximum weight (W) that can be carried by the thief is 20 pounds.
- There are 16 items in the store that the thief can take (n = 16).
- Their values and corresponding weights are defined by the following two lists:
- Item Values:
10, 5, 30, 8, 12, 30, 50, 10, 2, 10, 40, 80, 100, 25, 10, 5
. - Item Weights:
1, 4, 6, 2, 5, 10, 8, 3, 9, 1, 4, 2, 5, 8, 9, 1
.
- Item Values:
Instructions:
- Develop an algorithm using Java and developed in the Cloud9 environment (or your own Java IDE) environment to solve the knapsack problem.
- Your solution should be based upon dynamic programming principles as opposed to brute force.
- The optimal solution is one that is less than or equal to 20 pounds of weight and one that has the highest value.
Brute Force Algorithm:
- The brute force approach would be to look at every possible combination of items that is less than or equal to 20 pounds. We know that the brute force approach will need to consider every possible combination of items which is 2n items or 65536.
Table of Contents¶
- WA5. Knapsack Problem - Dynamic Programming
- Statement
- Table of Contents
- Introduction
- Source Code
- Source Code Explanation
- Output
- Complexity Analysis
- Conclusion
- References
Introduction¶
The knapsack problem involves finding the optimal combination of items to include in a knapsack, given a set of items with different weights and values, and a knapsack with a maximum weight capacity. The goal is to maximize the total value of carried items without exceeding the weight capacity of the knapsack (Shaffer, 2011).
The dynamic programming approach to the knapsack problem involves breaking down the problem into smaller subproblems and solving them only once; and then memoize the results to solve larger problems. The key idea is to build a table to store the maximum value that can be achieved with a given weight capacity and a subset of items. The table is filled iteratively, starting with the smallest subproblem and building up to the full problem (Hristakeva & Shrestha, 2005).
This assignment provides a source code in java, an explanation of the source code, and example output. To run the code, you need to compile the source code and then run it. All files should be passed with the attachments.
To run the code, you need to:
- Compile the java code using the command
javac Wa5.java
. - Run the compiled code using the command
java Wa5
.
Source Code¶
Below is the source code for the assignment, all in one file, to make easy to run and test. The main class is Wa5
which contains other helper classes and methods to implement the dynamic programming solution to the knapsack problem.
import java.util.ArrayList;
import java.util.List;
class Wa5 {
public static void main(String[] args) {
int[] weights = { 1, 4, 6, 2, 5, 10, 8, 3, 9, 1, 4, 2, 5, 8, 9, 1 }; // weights of items available to steal
int[] Values = { 10, 5, 30, 8, 12, 30, 50, 10, 2, 10, 40, 80, 100, 25, 10, 5 }; // values of items available to steal
int maxWeight = 20; // maximum weight that can be carried by the thief
int n = weights.length; // number of items available to steal
ShopItem[] shopItems = new ShopItem[n];
for (int i = 0; i < n; i++) {
shopItems[i] = new ShopItem(weights[i], Values[i]);
}
KnapSack k = new KnapSack(shopItems, maxWeight);
k.solve();
}
public static class KnapSack {
private ShopItem[] items; // items available to steal
private int maxWeight; // maximum weight that can be carried by the thief
private int[][] dp; // dynamic programming table
public KnapSack(ShopItem[] items, int maxWeight) {
this.items = items;
this.maxWeight = maxWeight;
this.dp = new int[items.length + 1][maxWeight + 1]; // initialize the dp table [n * W]
}
public void populateTable() {
for (int i = 1; i <= items.length; i++) { // loop through the items
for (int j = 1; j <= maxWeight; j++) { // loop through capacities
if (items[i - 1].weight <= j) { // if the weight of the item is less than or equal to the current capacity (j)
int aboveCell = dp[i - 1][j]; // value of the cell above the current cell
int remainingWeight = j - items[i - 1].weight; // remaining weight after including the item in the knapsack
int valueOfItem = items[i - 1].value; // value of the item
int valueOfItemPlusRemaining = dp[i - 1][remainingWeight] + valueOfItem; // value of the item plus the remaining weight
int cellValue = Math.max(aboveCell, valueOfItemPlusRemaining); // put the maximum value in the cell
dp[i][j] = cellValue; // set the value of the cell
} else {
// we can not include the item in the knapsack, so we just copy the value from the cell above
dp[i][j] = dp[i - 1][j];
}
}
}
}
public void reportResults() {
// let's print the dp table
System.out.println("DP Table:");
System.out.printf("%10s", "i/j"); // blank space for row header
// headers for the columns
for (int j = 0; j <= maxWeight; j++) {
System.out.printf("%8d", j);
}
System.out.println();
// rows
for (int i = 0; i <= items.length; i++) {
if (i == 0) {
System.out.printf("%10s", "0");
} else {
System.out.printf("%10s", " " + i);
}
for (int j = 0; j <= maxWeight; j++) {
System.out.printf("%8d", dp[i][j]);
}
System.out.println();
}
// We will try to report the items that were taken, and their total value.
// there are simpler and less expensive ways to do this, e.g. totalValue = dp[n][maxWeight]
// but we will use the most straightforward way
List<ShopItem> itemsTaken = new ArrayList<>();
int totalValue = 0;
int w = maxWeight;
for (int i = items.length; i > 0 && w > 0; i--) {
if (dp[i][w] != dp[i - 1][w]) { // if the value of the cell is not equal to the value of the cell above
itemsTaken.add(items[i - 1]);
totalValue += items[i - 1].value; // Adjust the totalValue.
w -= items[i - 1].weight; // track the remaining weight
}
}
System.out.println("\n\n\n");
System.out.println("Total value taken by thief: " + totalValue);
System.out.println("Items taken:");
for (ShopItem item : itemsTaken) {
System.out.println("\t - Weight: " + item.weight + ", Value: " + item.value);
}
}
public void solve() {
populateTable();
reportResults();
}
}
public static class ShopItem {
int weight;
int value;
public ShopItem(int weight, int value) {
this.weight = weight;
this.value = value;
}
}
}
Source Code Explanation¶
- The main class
Wa5
contains amain
method that orchestrates the solution to the knapsack problem along with two helper classes:KnapSack
andShopItem
. - The
ShopItem
class represents an item available in the store with two attributes:weight
andvalue
. - The
KnapSack
class represents the knapsack problem and contains the logic to solve it using dynamic programming principles. - The
KnapSack
class has the following attributes and methods:items
: an array ofShopItem
objects representing the items available in the store.maxWeight
: an integer representing the maximum weight that can be carried by the thief.dp
: a two-dimensional array to store the results of subproblems.populateTable()
: a method to fill the dynamic programming table with the maximum value that can be achieved with a given weight capacity and a subset of items according to method from (Hristakeva & Shrestha, 2005).reportResults()
: a method to print the dynamic programming table and report the items taken by the thief and their total value.
- The
main
method initializes the items available in the store, the maximum weight that can be carried by the thief, and creates an instance of theKnapSack
class to solve the problem.
Output¶
By running the code against the provided input, the following output is generated:
Image 1: Output of the code when run against the provided input. |
---|
![]() |
And here is the text output:
Total value taken by thief: 280
Items taken:
- Weight: 5, Value: 100
- Weight: 2, Value: 80
- Weight: 4, Value: 40
- Weight: 8, Value: 50
- Weight: 1, Value: 10
Below is the table filled by the algorithm according to the method from (Hristakeva & Shrestha, 2005):
Items Capacity | 0 | 1 | 2 | 3 | 4 | 5 | … | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 |
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
0 | 0 | 0 | 0 | 0 | 0 | 0 | … | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 |
1 | 0 | 10 | 10 | 10 | 10 | 10 | … | 10 | 10 | 10 | 10 | 10 | 10 | 10 | 10 |
2 | 0 | 10 | 10 | 10 | 10 | 15 | … | 15 | 15 | 15 | 15 | 15 | 15 | 15 | 15 |
3 | 0 | 10 | 10 | 10 | 10 | 15 | … | 45 | 45 | 45 | 45 | 45 | 45 | 45 | 45 |
4 | 0 | 10 | 10 | 18 | 18 | 18 | … | 53 | 53 | 53 | 53 | 53 | 53 | 53 | 53 |
5 | 0 | 10 | 10 | 18 | 18 | 18 | … | 53 | 60 | 60 | 60 | 60 | 65 | 65 | 65 |
6 | 0 | 10 | 10 | 18 | 18 | 18 | … | 53 | 60 | 60 | 60 | 70 | 70 | 78 | 78 |
7 | 0 | 10 | 10 | 18 | 18 | 18 | … | 68 | 80 | 90 | 90 | 98 | 98 | 98 | 102 |
8 | 0 | 10 | 10 | 18 | 20 | 20 | … | 70 | 80 | 90 | 90 | 98 | 100 | 100 | 108 |
9 | 0 | 10 | 10 | 18 | 20 | 20 | … | 70 | 80 | 90 | 90 | 98 | 100 | 100 | 108 |
10 | 0 | 10 | 20 | 20 | 28 | 30 | … | 80 | 80 | 90 | 100 | 100 | 108 | 110 | 110 |
11 | 0 | 10 | 20 | 20 | 40 | 50 | … | 100 | 110 | 110 | 118 | 120 | 120 | 130 | 140 |
12 | 0 | 10 | 80 | 90 | 100 | 100 | … | 160 | 170 | 180 | 190 | 190 | 198 | 200 | 200 |
13 | 0 | 10 | 80 | 90 | 100 | 100 | … | 240 | 240 | 248 | 250 | 250 | 260 | 270 | 280 |
14 | 0 | 10 | 80 | 90 | 100 | 100 | … | 240 | 240 | 248 | 250 | 250 | 260 | 270 | 280 |
15 | 0 | 10 | 80 | 90 | 100 | 100 | … | 240 | 240 | 248 | 250 | 250 | 260 | 270 | 280 |
16 | 0 | 10 | 80 | 90 | 100 | 105 | … | 240 | 245 | 248 | 253 | 255 | 260 | 270 | 280 |
Complexity Analysis¶
The dynamic programming approach has the advantage of being more efficient than the brute force approach, as it eliminates redundant computations caused by the un-used irrelevant combinations storing and reusing the results of subproblems.
The time complexity of the dynamic programming approach is O(nW)
, where n
is the number of items and W
is the weight capacity of the knapsack, making it more practical for larger problem instances. The space complexity is O(nW)
to store the two dimensional array that holds the results of the subproblems (Hristakeva & Shrestha, 2005).
This clear from our implementation, as we have two nested loops to fill the dynamic programming table (one loop for the items (n) and one loop for the capacities (W)). The space complexity is also proportional to the number of items and the weight capacity of the knapsack (an array of size n * W
). Hence, both time and space complexity are linear and proportional to the number of items and the weight capacity of the knapsack O(nW).
Conclusion¶
We have provided the source code, source code explanation example output, and asymptotic analysis. Here are our checks for the assignment requirements:
Requirement | Achieved ? | Comment |
---|---|---|
Was a java algorithm solution for the knapsack problem provided ? | ✅ | See the Source code section or the attached java file. |
Is the code documented to give the reader ? | ✅ | Code comments + see code explanation section. |
Does the java algorithm execute in the Java IDE environment? | ✅ | See attached output, and follow instructions to run it yourself. |
When executed does the algorithm produce the appropriate output? | ✅ | Output: value 280 . weight 20 . stolen items: 5, 2, 4, 8, 1 |
Does the assignment include an asymptotic analysis ? | ✅ | See Complexity analysis section. O(nW) for both space and time. |
References¶
- Hristakeva, M. & Shrestha, D. (2005). Different Approaches to Solve the 0/1 Knapsack Problem. Simpson College. Computer Science Department. https://micsymposium.org/mics_2005/papers/paper102.pdf
- Shaffer, C. (2011). A Practical Introduction to Data Structures and Algorithm Analysis. Blacksburg: Virginia. Tech. https://people.cs.vt.edu/shaffer/Book/JAVA3e20130328.pdf. hapter 16: Patterns of Algorithms.