S3-算法-动态规划算法【2021-02-05】

总目录:地址如下看总纲

https://www.jianshu.com/p/929ca9e209e8

1、动态规划算法介绍

1、动态规划(Dynamic Programming)算法的核心思想是:将大问题划分为小问题进行解决,从而一步步获取最优解的处理算法

2、动态规划算法与分治算法类似,其基本思想也是将待求解问题分解成若干个子问题,先求解子问题,然后从这些子问题的解得到原问题的解。

3、与分治法不同的是,适合于用动态规划求解的问题,经分解得到子问题往往不是互相独立的。 ( 即下一个子阶段的求解是建立在上一个子阶段的解的基础上,进行进一步的求解 )

4、动态规划可以通过 填表 的方式来逐步推进,得到最优解

2、动态规划算法最佳实践-背包问题

有一个背包,容量为4磅 , 现有如下物品


image.png

要求:
1、要求达到的目标为装入的背包的总价值最大,并且重量不超出
2、要求装入的物品不能重复

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

1、背包问题主要是指一个给定容量的背包、若干具有一定价值和重量的物品,如何选择物品放入背包使物品的价值最大。
2、其中问题分两种,01背包 和 完全背包(完全背包指的是:每种物品都有无限件可用)
3、这里的问题属于01背包,即每种类物品最多放一个。而无限背包可以转化为01背包。
4、算法的主要思想,利用动态规划来解决。每次遍历到的第i个物品,根据 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解释:就是现在月薪很低的你买不起贵的,你只能买得起和上个月月薪一样的东西...)
(4) 当 j>=w[i] 时: v[i][j] = max{v[i-1][j], v[i]+v[i-1][j-w[i]]} ; // 当 准备加入的新增的商品的容量小于等于当前背包的容量(其中max 内的参数分两个 ,比较出最大的一个 ,赋值 给 v[i][j]),具体解释:
v[i-1][ j ] : 既上一个单元格的装入的最大值(已有的值)
v[ i - 1 ] [ j - w[ i ] ] : 装入i-1商品,到剩余空间为 j-w[i] 的最大值
注:此二位数组可以看成 x轴和 y 轴, v[y] [x] 或者 v[i] [j],以后表格对应,x轴对应的是 重量(容量),y轴则是 物品编号


背包问题填表

4、思路分析(图分解)

(1)背包问题:有一个背包,容量为4磅 , 现有如下物品,
要求达到的目标为装入的背包的总价值最大,并且重量不超出。


价格表

(2)解决类似的问题可以分解成一个个的小问题进行解决,假设存在背包容量大小分为1,2,3,4的各种容量的背包(分配容量的规则为最小重量的整数倍):


image.png

(3)第一次填表:对于第一行(i=1), 目前只有吉他可以选择,所以
image.png

(4)第二次填表:对于第二行(i=2),目前存在吉他和音响可以选择,所以
image.png

(5)第三次填表:对于第三行(i=3),目前存在吉他和音响、电脑可以选择,所以


image.png

(★)带数推导公式案例:
  1. 案例一:v[1][1] = ? ,已知 : w[1] = 1 , j = 1 (已知数据从价格表中获得)
    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] (已知数据从价格表中获得)
    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),数组角标 0 到 3 ,归属 y(i) 轴坐标 v[ y] [ x ] ;容量(重量) 是 4 磅(个单位),默认一个单位不存在,角标从 0开始, 总的是 5 件(4 + 1),数组角标 0 到 4 ,归属 x(j)轴,坐标 v[ y ] [ x] 。

5、代码

package com.kk.algorithm.dynamic;
/*
 * @Description:    动态规划(背包问题)
 * @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};
        // 背包容量(总的或者部分),因为它是可变的,代码中体现
        int m = 4;
        // 物品个数
        int n = w.length;


        // 构建二位数组(表格)
        // v[i][j] 表示在第i个物品中能够装入容量为j的背包中的最大价值
        int[][] v = new int[n + 1][m + 1];
        // 为了记录放入商品的情况,我们定一个二维数组
        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;
        }

        // 根据思路,进行动态规划实现
        for (int i = 1; i < v.length; i++) {// 第一行不处理,因为默认为0
            for (int j = 1; j < v[0].length; j++) {// 第一列不处理,因为默认为0
                // 公式 w[i] > j 实现
                if (w[i - 1] > j) {// 因为 从1 开始,因此公式 w[i] 需改为 w[i-1]
                    v[i][j] = v[i - 1][j];
                } else {
                    // 故因:i 从 1 开始,公式做调整
                    // 原公式: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 因为数组从 0 开始,这边循环直接从 1 开始所以 -1;
                    // w[] 中 -1 也同上理

                    //为了记录商品存放到背包的情况,我们不能直接的使用上面的公式,需要使用if-else来体现公式
                    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;// 记录当前坐标为最优,用状态 1 表示
                    } else {
                        // 若新增不是最优,就和上个单元格一致
                        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) {// 坐标满足最优
                System.out.printf("第%d个商品放入到背包\n", i);
                j -= w[i-1]; //w[i-1],查找一个物品后 减去对应的重量
            }
            i--;// 查找一个物品后 -1
        }
    }
}

6、坐标

image.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,793评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,567评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,342评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,825评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,814评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,680评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,033评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,687评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,175评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,668评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,775评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,419评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,020评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,206评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,092评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,510评论 2 343

推荐阅读更多精彩内容