摘要:狀態(tài)轉(zhuǎn)移方程背包問題的狀態(tài)轉(zhuǎn)移方程是其中即表示前件物品恰放入一個(gè)容量為的背包可以獲得的最大價(jià)值。求解將哪些物品裝入背包可使這些物品的體積總和不超過背包容量,且價(jià)值總和最大。
01背包 01背包的概念
有N件物品和一個(gè)容量為V的背包。第i件物品的費(fèi)用是c[i],價(jià)值是w[i]。求解將哪些物品裝入背包可使價(jià)值總和最大。
從這個(gè)題目中可以看出,01背包的特點(diǎn)就是:每種物品僅有一件,可以選擇放或不放。
01背包問題的狀態(tài)轉(zhuǎn)移方程是:
f[i][v]=max{f[i-1][v],f[i-1][v-c[i]]+w[i]}
其中,即fi表示前i件物品恰放入一個(gè)容量為v的背包可以獲得的最大價(jià)值。
上述方程右邊可以理解為兩種情況,取最優(yōu)值.
情況一:第i件不放進(jìn)去,這時(shí)所得價(jià)值為:fi-1;
情況二:第i件放進(jìn)去,這時(shí)所得價(jià)值為:fi-1]+w[i];
情況二的意思是,如果第i件放進(jìn)去,那么在容量V-c[i]里就要放進(jìn)前i-1件物品.(引用自)
案例解析有編號(hào)分別為a,b,c,d,e的五件物品,它們的重量分別是2,2,6,5,4,它們的價(jià)值分別是6,3,5,4,6,現(xiàn)在給你個(gè)承重為10的背包,如何讓背包里裝入的物品具有最大的價(jià)值總和?
首先要明確這張表是至底向上,從左到右生成的.
只要你能通過找規(guī)律手工填寫出上面這張表就算理解了01背包的動(dòng)態(tài)規(guī)劃算法。
為了敘述方便,用e2單元格表示e行2列的單元格,這個(gè)單元格的意義是用來(lái)表示只有物品e時(shí),有個(gè)承重為2的背包,那么這個(gè)背包的最大價(jià)值是0,因?yàn)閑物品的重量是4,背包裝不了
對(duì)于d2單元格,表示只有物品e,d時(shí),承重為2的背包,所能裝入的最大價(jià)值,仍然是0,因?yàn)槲锲積,d都不是這個(gè)背包能裝的。(引用自)
var n = 5; //物品數(shù)量,該參數(shù)可以自行設(shè)定 var v = 10; //背包體積,該參數(shù)可自行設(shè)定 var volumArray = [4, 3, 5, 2, 5];//物品體積組成的數(shù)組,該參數(shù)可自行設(shè)定 var valueArray = [9, 6, 1, 4, 1];//物品價(jià)值組成的數(shù)組,該參數(shù)可自行設(shè)定 var tempArray = []; for (let a = 0; a < n; a++) { tempArray[a] = []; for (let b = 0; b < v; b++) { tempArray[a].push(0); } } for (let i = 0; i < n; i++) { for (let j = 0; j < v; j++) { tempArray[i][j] = i == 0 ? 0 : tempArray[i - 1][j]; if (i > 0 && j >= volumArray[i - 1]) { tempArray[i][j] = tempArray[i][j] >= tempArray[i - 1][j - volumArray[i - 1]] + valueArray[i - 1] ? tempArray[i][j] : tempArray[i - 1][j - volumArray[i - 1]] + valueArray[i - 1]; } } } console.log(tempArray[n - 1][v - 1]);完全背包 完全背包的概念
有N種物品和一個(gè)容量為V的背包,每種物品都有無(wú)限件可用。
第i種物品的體積是c,價(jià)值是w。求解將哪些物品裝入背包可使這些物品的體積總和不超過背包容量,且價(jià)值總和最大。
參考上述01背包的轉(zhuǎn)移方程,不難得出:
f[i][v]=max{f[i-1][v-k*c[i]]+k*w[i]|0<=k*c[i]<=v}
偽代碼實(shí)現(xiàn)如下:
for i=1...N for v=0...V for k=1...v/w[i] f[i][v] = max{f[i-1][v],f[i-1][v-k*w[i]]+k*c[i]};案例解析
給你六種面額1,5,10,20,50,100元的紙幣,假設(shè)每種幣值的數(shù)量都足夠多,編寫程序求組成N元的不同組合個(gè)數(shù).
function fn (all) { const arr = [1, 5, 10, 20, 50, 100], len = arr.length, res = []; for (let i = 0; i <= len; i++) { res[i] = []; res[i][0] = 1; } for (let j = 1; j <= all; j++) { res[0][j] = 0; } for (let i = 1; i <= len; i++) { for (let j = 1; j <= all; j++) { res[i][j] = 0; for (let k = 0; k <= j / arr[i - 1]; k++) { res[i][j] += res[i - 1][j - k * arr[i - 1]]; } } } return res[len][all]; }
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/84070.html
摘要:遺傳算法物競(jìng)天擇,適者生存,遺傳算法就是借鑒生物體自然選擇和自然遺傳機(jī)制的隨機(jī)搜索算法。隨機(jī)生成的基因適應(yīng)度的最大值隨機(jī)生成,適應(yīng)度函數(shù)計(jì)算種群中所有對(duì)象的適應(yīng)度及總和,并對(duì)超出的基因進(jìn)行懲罰。 此為《算法的樂趣》讀書筆記,我用javascript(ES6)重新實(shí)現(xiàn)算法。 遺傳算法 物競(jìng)天擇,適者生存,遺傳算法就是借鑒生物體自然選擇和自然遺傳機(jī)制的隨機(jī)搜索算法。算法的關(guān)鍵點(diǎn)有:基因的選...
摘要:動(dòng)態(tài)規(guī)劃背包士兵路徑復(fù)雜度談算法一定要考慮復(fù)雜度時(shí)間復(fù)雜度和空間復(fù)雜度時(shí)間復(fù)雜度計(jì)算機(jī)基本操作的次數(shù)匯編指令的條數(shù)尋址跳轉(zhuǎn)空間復(fù)雜度所需占用的內(nèi)存字節(jié)數(shù)兩者區(qū)別空間是可以返回利用的。 面試求職班一筆記 算法主要研究:時(shí)空復(fù)雜度 算法的特征: 有窮性, 確定性, 可行性, 可能沒有輸入,但一定有輸出 常用算法 窮舉法(eg:求N個(gè)數(shù)的全排列;8皇后問題) 減而治之(二分查找...
摘要:終止條件遞推公式遞歸的分類通過做大量的題,根據(jù)遞歸解決不同的問題,引申出來(lái)的幾種解決和思考的方式。我們通過層與層之間的計(jì)算關(guān)系用遞推公式表達(dá)出來(lái)做計(jì)算,經(jīng)過層層的遞歸,最終得到結(jié)果值。 showImg(https://segmentfault.com/img/remote/1460000019222330); 前言 幾個(gè)月之前就想寫這樣一篇文章分享給大家,由于自己有心而力不足,沒有把真...
摘要:是一個(gè)遺傳算法框架,這里是它的簡(jiǎn)介。最小化問題使用負(fù)值的的最大化問題用正值。略種群種群橫線個(gè)體。這個(gè)種群是直接使用和函數(shù)來(lái)初始化的。個(gè)體之間分布在網(wǎng)格中每個(gè)格子包含一個(gè)個(gè)體。調(diào)用將會(huì)返回一個(gè)種群,個(gè)體是使用兩個(gè)索引可獲得的。 DEAP是一個(gè)python遺傳算法框架,這里是它的簡(jiǎn)介。DEAP documentation今天整理一下DEAP的概覽,大體了解一下它的流程。初學(xué),不嚴(yán)謹(jǐn),僅作為...
閱讀 697·2021-11-22 09:34
閱讀 3830·2021-09-22 15:42
閱讀 1342·2021-09-03 10:28
閱讀 1081·2021-08-26 14:13
閱讀 1911·2019-08-29 15:41
閱讀 1437·2019-08-29 14:12
閱讀 3375·2019-08-26 18:36
閱讀 3318·2019-08-26 13:47