回顾DP背包问题

动态规划三个重要性质:

  • 最优子结构
  • 重叠子问题
  • 无后效性(在构造解空间时一定要考虑)

一. 0/1背包

问题描述

01背包:有n种物品与承重为m的背包。每种物品只有一件,每个物品都有对应的重量weight[i]与价值value[i],最大化背包价值。

对于背包问题,常见的做法是构建二维数组来构造最优解。以背包问题来说,我们的目标是通过每个物品价值value[i]和重量weight[i],来最大化背包价值,而背包容量实际上则是优化条件

形式化定义:设dp[i][j]为背包价值函数,dp[i][j]表示在“背包的重量为j时,取前i件物品,所能达到的最大价值”。

对于物品i,此时我们的背包有两种状态,选或不选

  1. 如果选择物品i装入背包,则需要背包留出weight[i]的重量来容纳物品i,那么若想让dp[i][j]最大,则只需让dp[i-1][j-weight[i]]最大(即前i-1件物品,使得背包容量为j-weight[i]时价值最大);
  2. 如果不选择物品i装入背包,则显然dp[i-1][j]最大,便能使得dp[i][j]最大化;

于是得到了0/1背包问题的状态转移方程:
dp[i][j]=max{d[i-1][j], dp[i-1][j-weight[i]]+value[i]}

int zeroOneBackpack(int value[], int weight[], int items, int volume) {
    /**
     * 0-1背包,二维数组实现
     * */
    int dp[items + 1][volume + 1];
    for (int i = 1; i <= items; ++i) {
        for (int j = 1; j <= volume; ++j) {
            if (weight[i] > j)
                dp[i][j] = dp[i - 1][j];
            else
                dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - weight[i]] + value[i]);
        }
    }
    return dp[items][volume];
}

采用二维数组实现的方式比较直观,但是内存消耗比较大,如果数据量稍微大一点,估计就爆内存了。于是产生了一种叫“滚动数组”的方式。

滚动数组的思想是因为二维数组中存储了很多的中间状态,最后有用的状态只有dp[items][:]这一行(也就是dp表的最后一行),滚动数组通过迭代的替换中间状态来构造最优解。通过输出每次dp[j]可以看出,一维数组就是对二维数组中的每次中间结果进行更新,最后保留最后一行。(如果内层循环j是从1开始,那么就影响了“无后效性”,相当于每个物品的数目就不是1了,这样问题就变成了完全背包问题,后续计算的结果会受到影响)

一维数组状态转移方程:
dp[j]=max{d[j], dp[j-weight[i]]+value[i]}

int zeroOneBackpack_one_diem(int value[], int weight[], int items, int volume) {
    /**
     * 0-1背包,滚动数组实现
     * */
    int dp[volume + 1];
    for (int i = 1; i <= items; i++) {
        for (int j = volume; j >= weight[i]; j--) {
            // 内存循环如果反了就变成完全背包了
            dp[j] = max(dp[j], dp[j - weight[i]] + value[i]);
        }
    }
    return dp[volume];
}

打印每轮dp[]结果如下:

当i = 1 : 2  2   2   2   2   2   2   2   2   2   2   2   
当i = 2 : 2  2   5   7   7   7   7   7   7   7   7   7   
当i = 3 : 2  3   5   7   8   10  10  10  10  10  10  10  
当i = 4 : 2  3   5   7   8   10  12  13  15  17  18  20  
当i = 5 : 2  4   6   7   9   11  12  14  16  17  19  21  

二. 完全背包

问题描述

完全背包:有n种物品与承重为m的背包。每种物品有无限多件,每个物品都有对应的重量weight[i]与价值value[i],最大化背包价值。

根据问题描述,背包能装多少就装多少,既一件物品可以放入背包中k个。
那么,可以得到完全背包的转移方程
dp[i][j] = max{dp[I][j], dp[i-1][j - k * weight[i]] + k * value[i]}, 0 <= k*weight[i] <= j

int completelyPackage(int value[], int weight[], int items, int volume) {
    /**
     * 完全背包,二维数组实现
     * 能装多少就装多少
     * */
    int dp[items + 1][volume + 1];
    for (int i = 1; i <= items; ++i) {
        for (int j = 1; j <= volume; j++) {
            for (int k = 1; k * weight[i] <= j; k++) {
                if (dp[i - 1][j - k * weight[i]] + k * value[i] > dp[i][j])
                    dp[i][j] = dp[i - 1][j - k * weight[i]] + k * value[i];
            }
        }
    }
    return dp[items][volume];
}

同样为了避免二维数组内存开销大的问题,采用一维数组求解方式类似0/1背包
一维数组状态转移方程:
dp[j]=max{dp[j], dp[j-weight[i]]+value[i]}

int completelyPackage_one_diem(int value[], int weight[], int items, int volume) {
    /***
     * 完全背包,滚动数组实现
     */
    int dp[volume + 1];
    for (int i = 1; i <= items; ++i) {
        for (int j = weight[i]; j <= volume; ++j) {
            if (dp[j - weight[i]] + value[i] > dp[j])
                dp[j] = dp[j - weight[i]] + value[i];
        }
    }
    return dp[volume];
}

三. 多重背包

问题描述

多重背包:有n种物品与承重为m的背包。每种物品有有限件num[i],每个物品都有对应的重量weight[i]与价值value[i],最大化背包价值。

多重背包多了一个限制条件,那就是每个物品数量为num[i]件;
那么这个问题完全可以当成是0/1背包问题,也就是将相同的num[i]件物品i看成价值跟重量相同的num[i]件不同的物品。

于是利用0/1背包的思路,得到状态转移方程
dp[j]=max{dp[j], dp[j-weight[i]]+value[I]}

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