-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathknapsack.js
114 lines (91 loc) · 2.79 KB
/
knapsack.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
/**
* @typeof {object} Item
* @param {number} weight
* @param {number} value
*/
/**
* Get the maximum value item indexes that can be carried by the knapsack
* @param {Item[]} items
* @param {number} capacity
* @return {number[]} An array of item indexes
* @time complexity: O(mn) where m is the size of items, n is the capacity
* @space complexity: O(mn) where m is the size of items, n is the capacity
*/
const knapsackMaximumItems = (items, capacity = 0) => {
if (!Array.isArray(items) || capacity <= 0) {
return [];
}
const numItems = items.length;
const dp = new Array(numItems + 1).fill(null).map(() => new Array());
for (let i = 0; i <= numItems; i++) {
for (let weight = 0; weight <= capacity; weight++) {
if (i === 0 || weight === 0) {
dp[i][weight] = 0;
continue;
}
const item = items[i - 1];
const prevWeight = dp[i - 1][weight];
if (item.weight <= weight) {
const newWeight = item.weight + dp[i - 1][weight - item.weight];
dp[i][weight] = Math.max(newWeight, prevWeight);
continue;
}
dp[i][weight] = prevWeight;
}
}
let itemIndex = numItems,
weight = capacity,
carryItem = [];
while (itemIndex > 0 && weight > 0) {
if (dp[itemIndex][weight] !== dp[itemIndex - 1][weight]) {
carryItem.push(itemIndex - 1);
weight -= dp[itemIndex - 1][weight];
}
itemIndex--;
}
return carryItem;
};
/**
* Get the maximum value that can be carried by the knapsack
* @param {Item[]} items
* @param {number} capacity
* @return {number} Maximum value
* @time complexity: O(mn) where m is the size of items, n is the capacity
* @space complexity: O(mn) where m is the size of items, n is the capacity
*/
const knapsackMaximumValue = (items, capacity = 0) => {
if (!Array.isArray(items) || capacity <= 0) {
return [];
}
const numItems = items.length;
const dp = new Array(numItems + 1).fill(null).map(() => new Array());
for (let i = 0; i <= numItems; i++) {
for (let weight = 0; weight <= capacity; weight++) {
if (i === 0 || weight === 0) {
dp[i][weight] = 0;
continue;
}
const item = items[i - 1];
const prevWeight = dp[i - 1][weight];
if (item.weight <= weight) {
const newWeight = item.value + dp[i - 1][weight - item.weight];
dp[i][weight] = Math.max(newWeight, prevWeight);
continue;
}
dp[i][weight] = prevWeight;
}
}
let itemIndex = numItems,
weight = capacity;
while (itemIndex > 0 && weight > 0) {
if (dp[itemIndex][weight] !== dp[itemIndex - 1][weight]) {
weight -= dp[itemIndex - 1][weight];
}
itemIndex--;
}
return dp[numItems][capacity];
};
export {
knapsackMaximumItems,
knapsackMaximumValue
};