背包问题动态规划解法

0 1 背包问题描述

有n个物品,它们有各自的体积和价值,现有给定容量的背包,如何让背包里装入的物品具有最大的价值总和?

为方便讲解和理解,下面讲述的例子均先用具体的数字代入,即:eg:number=4,capacity=8

i(物品编号) 1 2 3 4

w(体积) 2 3 4 5

v(价值) 3 4 5 6

解题思路

为描述方便,首先定义一些变量:Vi表示第i个物品的价值,Wi表示第i个物品的体积,定义V(i,j)为当前背包容量j,前i个物品的最佳组合对应的价值,同时将背包问题抽象化(X1,X2,...,Xn, 其中Xi取0或1,表示第i个物品选或不选)。

1、建立模型,即求max(V1X1+V2X2+....+VnXn)

2、寻找约束条件,W1X1+W2X2+…+WnXn<capacity;

3、寻找递推关系式,面对当前商品有两种可能性:

包的容量比该商品体积小,装不下,此时的价值与前i-1个的价值是一样的,即V(i,j)=V(i-1,j);

还有足够的容量可以装该商品,但装了也不一定达到当前最优价值,所以在装与不装之间选择最优的一个,即V(i,j)=max{V(i-1,j),V(i-1,j-w(i))+v(i)}。

其中V(i-1,j)表示不装,V(i-1,j-w(i))+v(i) 表示装了第i个商品,背包容量减少w(i),但价值增加了v(i);

由此可以得出递推关系式:

j<w(i)      V(i,j)=V(i-1,j)

j>=w(i)     V(i,j)=max{V(i-1,j),V(i-1,j-w(i))+v(i)}

解题代码

python 解法:

#!/bin/bash python

#coding:utf-8

def calMaxValue(w, v, c):

    """

    计算0,1背包最佳组合方案

    w: 物品的重量list

    v: 物品的价值list

    c: 背包重量

    """

    w_len = len(w)

    w.insert(0, 0)

    v.insert(0, 0)

    dp = [[0 for _ in range(c + 1)] for _ in range(w_len + 1)]

    for i in range(1, w_len + 1):

        for j in range(1, c + 1):

            if w[i] > j:

                dp[i][j] = dp[i - 1][j]

            else:

                dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - w[i]] + v[i])

    return dp[w_len][c]

w = [2, 3, 4, 5]

v = [3, 4, 5, 6]

c = 8

num = calMaxValue(w, v, c)

print(num)

带经过节点的写法:

#!/bin/bash python

#coding:utf-8

def calMaxValue(w, v, c):

    """

    计算0,1背包最佳组合方案

    w: 物品的重量list

    v: 物品的价值list

    c: 背包重量

    """

    w_len = len(w)

    w.insert(0, 0)

    v.insert(0, 0)

    dp = [[0 for _ in range(c + 1)] for _ in range(w_len + 1)]

    for i in range(1, w_len + 1):

        for j in range(1, c + 1):

            if w[i] > j:

                dp[i][j] = dp[i - 1][j]

            else:

                dp[i][j] = max(dp[i - 1][j], dp[i - 1][j - w[i]] + v[i])

    i = w_len

    j = c

    l = []

    while i > 0:

        if dp[i][j] == dp[i - 1][j]:

            i = i - 1

        elif w[i] <= j and dp[i][j] == dp[i - 1][j - w[i]] + v[i]:

            l.insert(0, i)

            j = j - w[i]

            i = i - 1

    print(l)   

    return dp[w_len][c]

w = [2, 3, 4, 5]

v = [3, 4, 5, 6]

c = 8

num = calMaxValue(w, v, c)

print(num)

参考文章:https://blog.csdn.net/qq_38410730/article/details/81667885

完全背包

完全背包,和0,1背包的区别在于,每件物品的数量无限,解法与0,1背包相似,推导公式要有一点变化,

V[i][j] = max(V[i -1][j], V[i][j - w[i]] + v[i]),因为放入一个i之后,还可以再放入i。

#!/bin/bash python

#coding:utf-8

def calMaxValue(w, v, c):

    """

    计算完全背包最佳组合方案

    w: 物品的重量list

    v: 物品的价值list

    c: 背包重量

    """

    w_len = len(w)

    w.insert(0, 0)

    v.insert(0, 0)

    dp = [[0 for _ in range(c + 1)] for _ in range(w_len + 1)]

    for i in range(1, w_len + 1):

        for j in range(1, c + 1):

            if w[i] > j:

                dp[i][j] = dp[i - 1][j]

            else:

                dp[i][j] = max(dp[i - 1][j], dp[i][j - w[i]] + v[i])

    i = w_len

    j = c

    res = {}

    while i > 0:

        # 循环取i,直到取不到为止

        while w[i] <= j and dp[i][j] == dp[i][j - w[i]] + v[i]:

            res[i] = res.get(i, 0) + 1

            j = j - w[i]

        i -= 1

    print(res)

    return dp[w_len][c]

w = [1, 3, 4, 7]

v = [2, 3, 5, 9]

c = 10

num = calMaxValue(w, v, c)

print(num)

多重背包

多重背包和完全背包,以及0,1背包的区别在于,多重背包中每个物品的个数都是给定的。

新的推导公式为V[i][j] = max(V[i-1][j], V[i - 1][j - k * w[i]] + k *v[i]),其中k为放了几次编号为i的物品。

多重背包还有种解法,是讲多重背包按数目展开,变成0,1背包问题。

#!/bin/bash python

#coding:utf-8

def calMaxValue(w, v, n, c):

    """

    计算多重背包最佳组合方案

    w: 物品的重量list

    v: 物品的价值list

    n: 物品的数量

    c: 背包重量

    """

    w_len = len(w)

    w.insert(0, 0)

    v.insert(0, 0)

    n.insert(0, 0)

    dp = [[0 for _ in range(c + 1)] for _ in range(w_len + 1)]

    for i in range(1, w_len + 1):

        for j in range(1, c + 1):

            if w[i] > j:

                dp[i][j] = dp[i - 1][j]

            else:

                count = min(n[i], j/w[i])

                dp[i][j] = dp[i - 1][j]

                for k in range(1, count + 1):

                    dp[i][j] = max(dp[i][j], dp[i - 1][j - k * w[i]] + k * v[i])

    print[dp]

    i = w_len

    j = c

    res = {}

    while i > 0:

        count = min(n[i], j / w[i])

        for k in range(count, 0, -1):

            # 去满足的最大k值

            if dp[i][j] == dp[i - 1][j - k * w[i]] + k * v[i]:

                res[i] = k

                j = j - k * w[i]

                break;

        i -= 1

    print(res)   

    return dp[w_len][c]

w = [1, 2, 2]

v = [6, 10, 20]

n = [10, 5, 2]

c = 8

num = calMaxValue(w, v, n, c)

print(num)

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

推荐阅读更多精彩内容