題目
有N種物品和一個(gè)容量為V的背包越庇。第i種物品最多有n[i]件可用,每件費(fèi)用是c[i],價(jià)值是w[i]斟薇。求解將哪些物品裝入背包可使這些物品的費(fèi)用總和不超過(guò)背包容量,且價(jià)值總和最大恕酸。
基本算法
這題目和完全背包問(wèn)題很類似堪滨。基本的方程只需將完全背包問(wèn)題的方程略微一改即可蕊温,因?yàn)閷?duì)于第i
種物品有n[i]+1
種策略:取0
件袱箱,取1
件……取n[i]
件遏乔。令f[i][v]
表示前i
種物品恰放入一個(gè)容量為v
的背包的最大權(quán)值,則有狀態(tài)轉(zhuǎn)移方程:
f[i][v]=max{f[i-1][v-k*c[i]]+k*w[i]|0<=k<=n[i]}```
復(fù)雜度是```O(V*Σn[i])```发笔。
####轉(zhuǎn)化為01背包問(wèn)題
另一種好想好寫(xiě)的基本方法是轉(zhuǎn)化為01背包求解:把第```i```種物品換成```n[i]```件01背包中的物品盟萨,則得到了物品數(shù)為```Σn[i]```的01背包問(wèn)題,直接求解了讨,復(fù)雜度仍然是```O(V*Σn[i])```捻激。
<p>但是我們期望將它轉(zhuǎn)化為01背包問(wèn)題之后能夠像完全背包一樣降低復(fù)雜度。仍然考慮二進(jìn)制的思想前计,我們考慮把第```i```種物品換成若干件物品胞谭,使得原問(wèn)題中第```i```種物品可取的每種策略——取```0..n[i]```件——均能等價(jià)于取若干件代換以后的物品。另外男杈,取超過(guò)```n[i]```件的策略必不能出現(xiàn)丈屹。
<p>方法是:將第```i```種物品分成若干件物品,其中每件物品有一個(gè)系數(shù)势就,這件物品的費(fèi)用和價(jià)值均是原來(lái)的費(fèi)用和價(jià)值乘以這個(gè)系數(shù)泉瞻。使這些系數(shù)分別為```1,2,4,...,2^(k-1),n[i]-2^k+1```,且```k```是滿足```n[i]-2^k+1>0```的最大整數(shù)苞冯。例如袖牙,如果```n[i]```為```13```,就將這種物品分成系數(shù)分別為```1,2,4,6```的四件物品舅锄。
<p>分成的這幾件物品的系數(shù)和為```n[i]```鞭达,表明不可能取多于```n[i]```件的第```i```種物品。另外這種方法也能保證對(duì)于```0..n[i]```間的每一個(gè)整數(shù)皇忿,均可以用若干個(gè)系數(shù)的和表示畴蹭,這個(gè)證明可以分```0..2^k-1```和```2^k..n[i]```兩段來(lái)分別討論得出,并不難鳍烁,希望你自己思考嘗試一下叨襟。
<p>這樣就將第```i```種物品分成了```O(log n[i])```種物品,將原問(wèn)題轉(zhuǎn)化為了復(fù)雜度為```O(V*Σlog n[i])```的01背包問(wèn)題幔荒,是很大的改進(jìn)糊闽。
下面給出```O(log amount)```時(shí)間處理一件多重背包中物品的過(guò)程,其中```amount```表示物品的數(shù)量:
procedure MultiplePack(cost,weight,amount)
if costamount>=V
CompletePack(cost,weight)
return
integer k=1
while k<num
ZeroOnePack(kcost,kweight)
amount=amount-k
k=k2
ZeroOnePack(amountcost,amountweight)```
希望你仔細(xì)體會(huì)這個(gè)偽代碼爹梁,如果不太理解的話右犹,不妨翻譯成程序代碼以后,單步執(zhí)行幾次姚垃,或者頭腦加紙筆模擬一下念链,就會(huì)慢慢理解了。
O(VN)的算法
多重背包問(wèn)題同樣有O(VN)的算法。這個(gè)算法基于基本算法的狀態(tài)轉(zhuǎn)移方程掂墓,但應(yīng)用單調(diào)隊(duì)列的方法使每個(gè)狀態(tài)的值可以以均攤O(1)的時(shí)間求解谦纱。由于用單調(diào)隊(duì)列優(yōu)化的DP已超出了NOIP的范圍,故本文不再展開(kāi)講解梆暮。我最初了解到這個(gè)方法是在樓天成的“男人八題”幻燈片上服协。
小結(jié)
這里我們看到了將一個(gè)算法的復(fù)雜度由O(VΣn[i])改進(jìn)到O(VΣlog n[i])的過(guò)程绍昂,還知道了存在應(yīng)用超出NOIP范圍的知識(shí)的O(VN)算法啦粹。希望你特別注意“拆分物品”的思想和方法,自己證明一下它的正確性窘游,并將完整的程序代碼寫(xiě)出來(lái)唠椭。