preface
Every time I see the backpack problem, I get a headache. Just summarize it.
Topic describes
There are n items, they have their own volume and value, existing backpack with a given capacity, how to make the items in the backpack have the maximum sum of value? For example, number = 4,capacity = 8
That’s two arrays
W = 5-tetrafluorobenzoic [2]
V =,4,5,6 [3]
There is a one-to-one correspondence between the two. For example, if the first item has a volume of 2, its value is 3.
The general idea of dynamic planning
Whether a problem can be written in dynamic programming depends on whether it can break down big problems into smaller ones, whether it satisfies the principle of optimality. If yes, then we can look for recursion between big problems and small problems. In my opinion, dynamic programming is a process of filling a table after finding the relationship. In the process of filling a table, each small cell is the current optimal solution, which is derived from the previous small problem, and finally the optimal solution of the big problem can be found.
Train of thought
To make the logic clearer, define some variables: v(I): represents the value of the ith item w(I): represents the volume of the ith item dp(I,j): J represents the current backpack capacity (we find the optimal solution by constantly changing the backpack capacity and the number of items), and DP (I,j) represents the maximum value of the previous I items. Now we need to find the recursion. We can assume that we know the optimal solution for dp(I,j), so the key is here, how to derive the recursion formula for the small problem. When you add an item, you either put it in or you don’t put it in. If you don’t put it in you get dp(I,j) = dp(I minus 1,j), so if you don’t put it in then the current optimal solution is the optimal solution for the last little problem. Dp (I,j) = dp(I -1,j-w[I]) +v[I], if not, we can deduce that the value of the current item must be dp(I -1,j-w[I]). If you do that then it becomes dp(I -1,j-w[I]) +v[I]. Dp (I,j) = Max (dp(I -1,j),dp(I -1,j-w[I])+v[I]) when adding an item, it may actually remove some of the items in front of it, and may be of lower value, so we take the larger of the two.
code
function knapsack(W,N,weight,value) {
let dp = new Array(N+1).fill(0).map(res= > new Array(W+1).fill(0))
for(let i = 1; i <= N; i++) {
let w = weight[i - 1]
let v = value[i-1]
for(let j = 0; j <= W; j++) {
if(w<=j) {
dp[i][j] = Math.max(dp[i-1][j], dp[i-1][j-w]+ v)
}
else {
dp[i][j] = dp[i-1][j]
}
}
}
return dp[N][W]
}
let weight = [2.3.4.5]
let value = [3.4.5.6]
console.log(knapsack(8.4,weight,value))
Copy the code
Print this array and we can see the table that’s filled in
Code optimization
We can actually see from the code above that the optimal solution for the first I items only depends on the first I minus 1, so we can actually solve the problem with a one-dimensional array, overwriting the values of the one-dimensional array each time. But the important thing to note here is that we have to iterate backwards, because the value behind the one-dimensional array is inferred from the previous value, and if we change the previous value in the first place, the code will have a problem. The code is as follows:
function k1(W,N,weight,value) {
let dp = new Array(W + 1).fill(0)
for (let i = 1; i <=N; i++) {
let w = weight[i - 1]
let v = value[i-1]
for(let j = W; j >=0; j--) {
if(w<=j)dp[j] = Math.max(dp[j],dp[j-w] + v)
}
}
return dp[W]
}
let weight = [2.3.4.5]
let value = [3.4.5.6]
console.log(k1(8.4,weight,value))
Copy the code
Again, print an array
conclusion
The key to dynamic programming is how to push small problems out of big problems rather than big problems out of small problems. Oneself still is the small white of dynamic planning, if have write incorrect place to still ask great understanding 🙃!