怎么输出一个二维数组_图解算法怎么用动态规划解决01背包问题

问题描述

对于一组不同重量、不可分割的物品,我们需要选择一些装入背包,在满足背包最大重量限制前提下,背包中物品总重量的最大值是多少?假设此时是5个物品,2,2,4,6,3,然后背包最大承载两是9.

假如我们使用回溯算法解决该问题, 代码如下

int maxW = 0; //最大重量

int n = 5; //物品数目

int w = 9; // 背包最大重量

int weight[] = {2,2,4,6,3};// 物品重量,2,2,4,6,3

void rucksack(int i, int cw)

{

if ( cw == w || i == n ){

if ( cw > maxW ) maxW= w;

return ;

}

rucksack(i + 1, cw);//不装第i个物品

if ( cw + weight[i] <= w){ //如果装的下

rucksack(i + 1, cw + weight[i]);

}

}

如果将代码执行过程产生状态画成树,我们可以发现对于不加入物品2的选择f(1,0)和加入物品2的选择f(1,2), 在下一个选择时,他们有一个相同的状态,f(2,2)。

c671b22f8d51af6f44ec78a768ee8cfb.png

虽然得到它的过程不同,但是从这往后,大家都是一样的。如同你走迷宫,走到一个地方,你发现墙上有一张纸写着,"此路不通",你就可以不用白费力气去探索了。因此,如果一个问题解决时存在重复子问题,我们可以通过记忆化的方式,避免重复运算,提高计算效率。

从动态规划的角度,我们可以将整个求解过程分为n个阶段,每个阶段都需要决策是否需要将物品放到背包中。每个物品的决策后,背包中物品的重量就有会有种情况,也就是达到了不同的状态,也就是递归树中的不同节点。在每一个层中,我们只记录不同的状态(比如说上图的第2层的两个 f(2,2)就可以合并成一种情况,当然第四层就更多了)。这样一来,我们就保证了每一层的状态数就不会超过w个(w是背包的承载重量)。这种合并操作就可以认为是一种记忆化。

我们先用一个二维数组来记录每层可以达到的不同状态

41bbffca8c085d59c7514314e0867028.png

考场重量为2的物品后,会出现两种状态,0和2。

918b76c97a4143513b0eb0b8a70e276b.png

在上一个状态的基础再考察一个重量为2的物品,会有三种状态,一直不选择,先不选再选2,两次都选2.

58744ed76e4d0cc89a6c783cc06afa79.png

继续考察重量为4的情况时,会出现五种情况,其中重量为4可能来源是0+0+4,2+2+0,这种重复状态就被合并成一种状态,因此减少了计算量。

7937ad11392c4b11c10b806a0913866d.png

最终状态如下

78687f3ea5e86d0d7cf5a531b14f554e.png

将上面的思考过程翻译成代码就是,

int rucksackDp(int *weight, int num, int w)

{

//先定义一个二维数组

int **status = (int **)malloc( sizeof(int *) * num);

//初始化数组

for (int i = 0; i < num; i++){

status[i] = (int *)calloc( w+1, sizeof(int));

}

//初始化第一行的值

status[0][0] = 1; //不放

if ( weight[0] < w){

status[0][weight[0]] = 1; //放

}

//从第二个开始考虑

for (int i = 1; i < num; i++){

//不放的情况,直接将上面的结果复制给当前

for (int j = 0; j <= w; j++){

if (status[i-1][j] == 1) status[i][j] = status[i-1][j];

}

//上面循环可以写成

//memcpy(status[i], status[i-1], sizeof(int) * num);

//放: 在上一个状态基础上, 将增加后的重量对应位置设置为1

for ( int j = 0; j <= w -weight[i]; j++){

if ( status[i-1][j] == 1) status[i][j+weight[i]] = 1;

}

}

//输出结果

for ( int i = w; i >= 0; --i){

if (status[n-1][i] == 1) return i;

}

return 0;

}

上面我们使用的是二维数组用于保存所有状态,但实际上这里我们只需要一维数组维护前一个状态就可以推导出当前结果

int rucksackDp2(int *weight, int num, int w)

{

//先定义数组

int *status = (int *)calloc( num, sizeof(int ) );

//初始化第一行的值

status[0]= 1; //不放

if ( weight[0] < w){

status[weight[0]] = 1; //放

}

//从第二个开始考虑

for (int i = 1; i < num; i++){

//不放: 保持状态不变

//放: 在上一个状态基础上, 将增加后的重量对应位置设置为1

for ( int j = 0; j <= w -weight[i]; j++){

if ( status[j] == 1) status[j+weight[i]] = 1;

}

}

//输出结果

for ( int i = w; i >= 0; --i){

if (status[i] == 1) return i;

}

return 0;

}

写动态规划代码的关键在于状态定义和状态转移方程。在0-1背包问题中,我们定义的状态是 status[i]就是当前决策结束后到达的重量,而转移方程就是 if(status[j]==1)status[j+weight[i]]=1;

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值