Recursion and Dynamic Programming
0/1 Knapsack (Small)
Fill a small 0/1 knapsack table where each row decides whether one more item is available.
Algorithm
@steps
- Create a table with one extra row for using zero items.
- Process the four items in fixed order.
- For each capacity, inherit when the item is too heavy.
- Otherwise compare
skipandtakefrom the previous row. - Print the best value and the full deterministic table. @end @complexity
- Time: O(item_count * capacity)
- Space: O(item_count * capacity) @end
state transition
`dp[i][w]` compares skipping item `i` with taking it and reading the remaining capacity from the previous row.
Go DSA Implementation
basic.go
package main
import (
"fmt"
"strings"
)
func rowString(row []int) string {
parts := make([]string, len(row))
for i, value := range row {
parts[i] = fmt.Sprintf("%d", value)
}
return "[" + strings.Join(parts, ", ") + "]"
}
func tableString(table [][]int) string {
parts := make([]string, len(table))
for i, row := range table {
parts[i] = rowString(row)
}
return "[" + strings.Join(parts, ", ") + "]"
}
func main() {
weights := []int{2, 3, 4, 5}
values := []int{3, 4, 5, 6}
capacity := 5
dp := make([][]int, len(weights) + 1)
for i := range dp {
dp[i] = make([]int, capacity + 1)
}
for item := 1; item <= len(weights); item++ {
weight := weights[item - 1]
value := values[item - 1]
for cap := 0; cap <= capacity; cap++ {
if weight > cap {
dp[item][cap] = dp[item - 1][cap]
} else {
skip := dp[item - 1][cap]
take := value + dp[item - 1][cap - weight]
if take > skip { dp[item][cap] = take } else { dp[item][cap] = skip }
}
}
}
fmt.Println(dp[len(weights)][capacity])
fmt.Println(tableString(dp))
}
@end @output 7 [[0, 0, 0, 0, 0, 0], [0, 0, 3, 3, 3, 3], [0, 0, 3, 4, 4, 7], [0, 0, 3, 4, 5, 7], [0, 0, 3, 4, 5, 7]] @end