S3-算法-動態(tài)規(guī)劃算法【2021-02-05】

總目錄:地址如下看總綱

http://www.reibang.com/p/929ca9e209e8

1惋嚎、動態(tài)規(guī)劃算法介紹

1继找、動態(tài)規(guī)劃(Dynamic Programming)算法的核心思想是:將大問題劃分為小問題進行解決,從而一步步獲取最優(yōu)解的處理算法

2袱结、動態(tài)規(guī)劃算法與分治算法類似,其基本思想也是將待求解問題分解成若干個子問題箍镜,先求解子問題剥懒,然后從這些子問題的解得到原問題的解虫溜。

3、與分治法不同的是,適合于用動態(tài)規(guī)劃求解的問題耐朴,經(jīng)分解得到子問題往往不是互相獨立的借卧。 ( 即下一個子階段的求解是建立在上一個子階段的解的基礎(chǔ)上,進行進一步的求解 )

4筛峭、動態(tài)規(guī)劃可以通過 填表 的方式來逐步推進铐刘,得到最優(yōu)解

2、動態(tài)規(guī)劃算法最佳實踐-背包問題

有一個背包影晓,容量為4磅 镰吵, 現(xiàn)有如下物品


image.png

要求:
1、要求達到的目標為裝入的背包的總價值最大挂签,并且重量不超出
2疤祭、要求裝入的物品不能重復

3、思路分析(文字說明)

1饵婆、背包問題主要是指一個給定容量的背包勺馆、若干具有一定價值和重量的物品,如何選擇物品放入背包使物品的價值最大侨核。
2谓传、其中問題分兩種,01背包 和 完全背包(完全背包指的是:每種物品都有無限件可用)
3芹关、這里的問題屬于01背包续挟,即每種類物品最多放一個。而無限背包可以轉(zhuǎn)化為01背包侥衬。
4诗祸、算法的主要思想,利用動態(tài)規(guī)劃來解決轴总。每次遍歷到的第i個物品直颅,根據(jù) w[i] 和 v[i] 來確定是否需要將該物品放入背包中。v[i][j] 表示在前 i 個物品中能夠裝入容量為 j 的背包中的最大價值怀樟,公式和解釋如下:
(1) w[i] 表示 第 i 個商品的重量功偿, v[i] 表示 第 i 個商品的價值(價格) ,j 表示 當前背包的容量
(2) v[i][0] = v[0][j] =0; //表示 填入表 第一行和第一列是0
(3) 當w[i]> j 時:v[i][j] = v[i-1][j] ; // 當準備加入新增的商品的容量大于 當前背包的容量時往堡,就直接使用上一個單元格的裝入策略 (阿K解釋:就是現(xiàn)在月薪很低的你買不起貴的械荷,你只能買得起和上個月月薪一樣的東西...)
(4) 當 j>=w[i] 時: v[i][j] = max{v[i-1][j], v[i]+v[i-1][j-w[i]]} ; // 當 準備加入的新增的商品的容量小于等于當前背包的容量(其中max 內(nèi)的參數(shù)分兩個 ,比較出最大的一個 虑灰,賦值 給 v[i][j]),具體解釋:
v[i-1][ j ] : 既上一個單元格的裝入的最大值(已有的值)
v[ i - 1 ] [ j - w[ i ] ] : 裝入i-1商品吨瞎,到剩余空間為 j-w[i] 的最大值
注:此二位數(shù)組可以看成 x軸和 y 軸, v[y] [x] 或者 v[i] [j]穆咐,以后表格對應颤诀,x軸對應的是 重量(容量)字旭,y軸則是 物品編號


背包問題填表

4、思路分析(圖分解)

(1)背包問題:有一個背包崖叫,容量為4磅 遗淳, 現(xiàn)有如下物品,
要求達到的目標為裝入的背包的總價值最大心傀,并且重量不超出洲脂。


價格表

(2)解決類似的問題可以分解成一個個的小問題進行解決,假設存在背包容量大小分為1剧包,2恐锦,3,4的各種容量的背包(分配容量的規(guī)則為最小重量的整數(shù)倍):


image.png

(3)第一次填表:對于第一行(i=1), 目前只有吉他可以選擇疆液,所以
image.png

(4)第二次填表:對于第二行(i=2),目前存在吉他和音響可以選擇,所以
image.png

(5)第三次填表:對于第三行(i=3),目前存在吉他和音響一铅、電腦可以選擇,所以


image.png

(★)帶數(shù)推導公式案例:
  1. 案例一:v[1][1] = ? ,已知 : w[1] = 1 堕油, j = 1 (已知數(shù)據(jù)從價格表中獲得)
    v[i][j]=max{v[i-1][j],v[i-1][j-w[i]]+v[i]} = v[1][1] = max{v[0][1], v[0][0]+v[1]} = max{0, 0 + 1500}潘飘,其中v[1]是第一件物品 吉他的價格
  2. 案例二:v[3][4] = ? ,已知: w[3] = 3 掉缺, j >= w[3] (已知數(shù)據(jù)從價格表中獲得)
    v[i][j]=max{v[i-1][j],v[i-1][j-w[i]]+v[i]} = v[3][4] = max{v[2][4],v[2][1]+v[3] = max{3000,1500+3000}卜录,其中v[1]是第三件物品 電腦的價格

★注意:看表格的時候,記得物品是3 件眶明,默認添加一件 不存在的 艰毒,角標從0開始,總的是 4件(3+1)搜囱,數(shù)組角標 0 到 3 丑瞧,歸屬 y(i) 軸坐標 v[ y] [ x ] ;容量(重量) 是 4 磅(個單位)蜀肘,默認一個單位不存在绊汹,角標從 0開始, 總的是 5 件(4 + 1)扮宠,數(shù)組角標 0 到 4 西乖,歸屬 x(j)軸,坐標 v[ y ] [ x] 坛增。

5获雕、代碼

package com.kk.algorithm.dynamic;
/*
 * @Description:    動態(tài)規(guī)劃(背包問題)
 * @Author:         Jk_kang
 * @CreateDate:     2021/2/5 0:25
 * @Param:
 * @Return:
**/
public class KnapsackProblem {
    public static void main(String[] args) {

        // 物品的重量
        int[] w = {1, 4, 3};
        // 物品的單價(價值),此處的val[i] 既 v[i] ,某個物品的單價
        int[] val = {1500, 3000, 2000};
        // 背包容量(總的或者部分)轿偎,因為它是可變的典鸡,代碼中體現(xiàn)
        int m = 4;
        // 物品個數(shù)
        int n = w.length;


        // 構(gòu)建二位數(shù)組(表格)
        // v[i][j] 表示在第i個物品中能夠裝入容量為j的背包中的最大價值
        int[][] v = new int[n + 1][m + 1];
        // 為了記錄放入商品的情況被廓,我們定一個二維數(shù)組
        int[][] table = new int[n + 1][m + 1];

        // 初始第一行坏晦,第一列,默認為 0 可以不設置,但是我愿意昆婿!
        for (int i = 0; i < v.length; i++) {
            // 初始第一列
            v[i][0] = 0;
        }
        for (int i = 0; i < v[0].length; i++) {
            // 初始第一列
            v[0][i] = 0;
        }

        // 根據(jù)思路球碉,進行動態(tài)規(guī)劃實現(xiàn)
        for (int i = 1; i < v.length; i++) {// 第一行不處理,因為默認為0
            for (int j = 1; j < v[0].length; j++) {// 第一列不處理仓蛆,因為默認為0
                // 公式 w[i] > j 實現(xiàn)
                if (w[i - 1] > j) {// 因為 從1 開始睁冬,因此公式 w[i] 需改為 w[i-1]
                    v[i][j] = v[i - 1][j];
                } else {
                    // 故因:i 從 1 開始,公式做調(diào)整
                    // 原公式:v[i][j] = Math.max (v[i - 1][j], v[i] + v[i - 1][j - w[i]]);
                    // 修改為:v[i][j] = Math.max (v[i - 1][j], val[i - 1] + v[i - 1][j - w[i - 1]]);
                    // val[] 中 -1 因為數(shù)組從 0 開始看疙,這邊循環(huán)直接從 1 開始所以 -1豆拨;
                    // w[] 中 -1 也同上理

                    //為了記錄商品存放到背包的情況,我們不能直接的使用上面的公式能庆,需要使用if-else來體現(xiàn)公式
                    if (v[i - 1][j] < val[i - 1] + v[i - 1][j - w[i - 1]]) {// 如果上一個單元格施禾,小于最新添加的剩余最大值
                        v[i][j] = val[i - 1] + v[i - 1][j - w[i - 1]];
                        // 則把情況記錄
                        table[i][j] = 1;// 記錄當前坐標為最優(yōu),用狀態(tài) 1 表示
                    } else {
                        // 若新增不是最優(yōu)搁胆,就和上個單元格一致
                        v[i][j] = v[i - 1][j];
                    }
                }
            }
        }

        //輸出一下v 看看目前的情況
        for (int i = 0; i < v.length; i++) {
            for (int j = 0; j < v[i].length; j++) {
                System.out.print (v[i][j] + "\t");
            }
            System.out.println ( );
        }

        //錯誤輸出, 這樣輸出會把所有的放入情況都得到, 其實我們只需要最后的放入
//      for(int i = 0; i < table.length; i++) {
//          for(int j=0; j < table[i].length; j++) {
//              if(table[i][j] == 1) {
//                  System.out.printf("第%d個商品放入到背包\n", i);
//              }
//          }
//      }

        //正確輸出:逆向輸出最后一個商品的 排列順序
        int i = table.length - 1; //行的最大下標
        int j = table[0].length - 1;  //列的最大下標
        while(i > 0 && j > 0 ) { //從table 的最后開始找
            if(table[i][j] == 1) {// 坐標滿足最優(yōu)
                System.out.printf("第%d個商品放入到背包\n", i);
                j -= w[i-1]; //w[i-1]弥搞,查找一個物品后 減去對應的重量
            }
            i--;// 查找一個物品后 -1
        }
    }
}

6、坐標

image.png
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末渠旁,一起剝皮案震驚了整個濱河市攀例,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌顾腊,老刑警劉巖粤铭,帶你破解...
    沈念sama閱讀 219,039評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異杂靶,居然都是意外死亡承耿,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,426評論 3 395
  • 文/潘曉璐 我一進店門伪煤,熙熙樓的掌柜王于貴愁眉苦臉地迎上來加袋,“玉大人,你說我怎么就攤上這事抱既≈吧眨” “怎么了?”我有些...
    開封第一講書人閱讀 165,417評論 0 356
  • 文/不壞的土叔 我叫張陵防泵,是天一觀的道長蚀之。 經(jīng)常有香客問我,道長捷泞,這世上最難降的妖魔是什么足删? 我笑而不...
    開封第一講書人閱讀 58,868評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮锁右,結(jié)果婚禮上失受,老公的妹妹穿的比我還像新娘讶泰。我一直安慰自己,他們只是感情好拂到,可當我...
    茶點故事閱讀 67,892評論 6 392
  • 文/花漫 我一把揭開白布痪署。 她就那樣靜靜地躺著,像睡著了一般兄旬。 火紅的嫁衣襯著肌膚如雪狼犯。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,692評論 1 305
  • 那天领铐,我揣著相機與錄音悯森,去河邊找鬼。 笑死绪撵,一個胖子當著我的面吹牛呐馆,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播莲兢,決...
    沈念sama閱讀 40,416評論 3 419
  • 文/蒼蘭香墨 我猛地睜開眼汹来,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了改艇?” 一聲冷哼從身側(cè)響起收班,我...
    開封第一講書人閱讀 39,326評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎谒兄,沒想到半個月后摔桦,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,782評論 1 316
  • 正文 獨居荒郊野嶺守林人離奇死亡承疲,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,957評論 3 337
  • 正文 我和宋清朗相戀三年邻耕,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片燕鸽。...
    茶點故事閱讀 40,102評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡兄世,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出啊研,到底是詐尸還是另有隱情御滩,我是刑警寧澤,帶...
    沈念sama閱讀 35,790評論 5 346
  • 正文 年R本政府宣布党远,位于F島的核電站削解,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏沟娱。R本人自食惡果不足惜氛驮,卻給世界環(huán)境...
    茶點故事閱讀 41,442評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望济似。 院中可真熱鬧矫废,春花似錦盏缤、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,996評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽娩脾。三九已至赵誓,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間柿赊,已是汗流浹背俩功。 一陣腳步聲響...
    開封第一講書人閱讀 33,113評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留碰声,地道東北人诡蜓。 一個月前我還...
    沈念sama閱讀 48,332評論 3 373
  • 正文 我出身青樓,卻偏偏與公主長得像胰挑,于是被迫代替她去往敵國和親蔓罚。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 45,044評論 2 355

推薦閱讀更多精彩內(nèi)容