01揹包問題
有 n 件物品和乙個容量是 v 的揹包。每件物品只能使用一次。
第 i 件物品的體積是 vi,價值是 wi。
求解將哪些物品裝入揹包,可使這些物品的總體積不超過揹包容量,且總價值最大。
輸出最大價值。
輸入格式
第一行兩個整數,n,v,用空格隔開,分別表示物品數量和揹包容積。
接下來有 n 行,每行兩個整數 vi,wi,用空格隔開,分別表示第 i 件物品的體積和價值。
輸出格式
輸出乙個整數,表示最大價值。
資料範圍
0原題鏈結
問題詳解1
問題詳解2
乙個超讚的思路
一維動態規劃
#include
#include
#include
using
namespace std;
const
int n=
1010
;int f[n]
;//表示前i個物品,當前全部體積是j的情況下,總價值是多少
int v[n]
,w[n]
;//體積用v價值用w表示
int m,n;
intmain()
二維動態規劃
#include
#include
#include
using
namespace std;
const
int n=
1010
;int f[n]
[n];
//表示前i個物品,當前全部體積是j的情況下,總價值是多少
int v[n]
,w[n]
;//體積用v價值用w表示
int m,n;
intmain()
/*int res=0;
for(int i=0;i<=m;i++)
res=max(res,f[n][i]);
cout
[m]<
//表示從1-n中選,總體積不超過m的集合
全新講解
//無優化版
#include
using
namespace std;
const
int n =
1010
;int n, m;
int v[n]
, w[n]
;int f[n]
[n];
intmain()
} cout << f[n]
[m]<< endl;
return0;
}/*i=1;v=1;w=2; j>=1;
f[1][0]: 0
f[1][1]: 0 f[1][1]: 2
f[1][2]: 0 f[1][2]: 2
f[1][3]: 0 f[1][3]: 2
f[1][4]: 0 f[1][4]: 2
f[1][5]: 0 f[1][5]: 2
i=2;v=2;w=4;j>=2;
f[2][0]: 0
f[2][1]: 2
f[2][2]: 2 f[2][2]: 4
f[2][3]: 2 f[2][3]: 6
f[2][4]: 2 f[2][4]: 6
f[2][5]: 2 f[2][5]: 6
i=3;v=3;w=4;j>=3;
f[3][0]: 0
f[3][1]: 2
f[3][2]: 4
f[3][3]: 6 f[3][3]: 6
f[3][4]: 6 f[3][4]: 6
f[3][5]: 6 f[3][5]: 8
i=4;v=4;w=5;j>=4;
f[4][0]: 0
f[4][1]: 2
f[4][2]: 4
f[4][3]: 6
f[4][4]: 6 f[4][4]: 6
f[4][5]: 8 f[4][5]: 8
*///有優化版
/*1. f[i] 僅用到了f[i-1]層,
2. j與j-v[i] 均小於j
3.若用到上一層的狀態時,從大到小列舉, 反之從小到大哦
*//*
#include using namespace std;
const int n = 1010;
int n, m;
int v[n], w[n];
int f[n];
int main()
*//*
f[j]代表的是f[i-1][j],也就是沒選現在這個,只從i-1個裡面選了
f[j-v[i]]+w[i]代表f[i-1][j-v[i]]+w[i],代表選了現在這個,並且與沒選的時候比較
i=1; j=5,j>=1,j--;
f[5]: 2
f[4]: 2
f[3]: 2
f[2]: 2
f[1]: 2
i=2;j>=2;
f[5]: 6
f[4]: 6
f[3]: 6
f[2]: 4
i=3; j>=3; v=4;w=4
f[5]=max(f[5],f[2]+4)=max(6,8): 8
f[4]=max(f[4],f[1]+4)=max(6,6): 6
f[3]=max(f[3],f[0]+4)=max(6,4): 6
i=4;j>=4;v=4,w=5
f[5]=max(f[5],f[1]+5)=max(8,7): 8
f[4]=max(f[4],f[0]+5)=max(6,5): 6
*/
為啥要逆序?a2:逆序是保證 dp[j - item_w[i]]的資料是上一層的,如果是正序你想想,j-item_w[i]肯定比j小啊,那麼他會被先遍歷到啊,那他就被更新了,就是本層的資料,就不是咱們要的上一層資料了,所以咱們要進行逆序
假如列舉到:i = 3, j = 8, v[3]
= 5, w[3]
= 1二維:dp[3]
[8]= max(dp[2]
[8], dp[2]
[3] + w[3]
) 此時的dp[2]
[8]和dp[2]
[3]都是上一輪的狀態值
一維:dp[8]
= max(dp[8], dp[3] + w[3]
) 我們要保證dp[8]和dp[3]都是上一輪的狀態值
按照逆序的順序,一維dp陣列的更新順序為:dp[8], dp[7], dp[6], ... , dp[3]
也就是說,在本輪更新的值,不會影響本輪中其他未更新的值!較小的index對應的狀態是上一輪的狀態值!
如果按照順序進行更新,dp[3]
= max(dp[3], dp[0] + w[0]
),對dp[3]的狀態進行了更新,那麼在更新dp[8]時,用到的dp[3]
就不是上一輪的狀態了,不滿足動態規劃的要求。
揹包問題 01揹包問題
n個物品,總體積是v,每個物品的體積的vi,每個物品的最大價值是wi,在不超過v的體積下求最大價值 eg揹包容積為 5 物品數量為 4 物品的體積分別為 物品的價值分別為 思路定義乙個二位陣列int f new int n 1 v 1 f i j 就表示在1 i個物品中選取體積小於v的情況的最大價值...
揹包問題 01揹包
有n件物品和乙個容量為v的揹包。第i件物品的重量是c i 價值是w i 求解將哪些物品裝入揹包可使價值總和最大。01揹包中的 01 就是一種物品只有1件,你可以選擇放進去揹包即1,也可以選擇不放入揹包中即0。include include using namespace std const int ...
揹包問題(01揹包)
1085 揹包問題 在n件物品取出若干件放在容量為w的揹包裡,每件物品的體積為w1,w2 wn wi為整數 與之相對應的價值為p1,p2 pn pi為整數 求揹包能夠容納的最大價值。input 第1行,2個整數,n和w中間用空格隔開。n為物品的數量,w為揹包的容量。1 n 100,1 w 10000...