【POJ】1837 Balance 【背包问题】

1 题解

设m表秤刻度的个数,n表示秤砣的数目。
数组p存储挂钩的位置,数组w存储秤砣的质量。
d[i][j]存储前i个秤砣平衡度为j的最大组合数(平衡度j>0表示左边重,j<0右边重,j=0平衡)。
利用动态规划状态方程思想:

d[i][j] += d[i-1][j-p[k]*w[i]]

前i个平衡度为j的数目 = 所有前i-1个,平衡度为 j-p[k]*w[i] 的数目(k取1~m)。
这样才能取到最大。
因为考虑到左右正负数的问题,所以所有j都加7500(20×25×15),此时平衡时j=7500。
最后d[n][7500]即为所求。

2 源代码

#include <iostream>
using namespace std;

int d[21][15002];
int p[21];          //position
int w[21];          //weight
int main()
{
    int i, j, k, m, n;
    while(cin>>m>>n){
        for(i=1; i<=m; i++)
            cin>>p[i];
        for(i=1; i<=n; i++)
            cin>>w[i];
        d[0][7500] = 1;
        for(i=1; i<=n; i++)
            for(j=-7500; j<=7500; j++)
                for(k=1; k<=m; k++)
                    if(j+7500>=p[k]*w[i])
                        d[i][j+7500] += d[i-1][j+7500-p[k]*w[i]];
        cout<<d[n][7500]<<endl;
    }
}


两维数组

Author: visaya fan <visayafan[AT]gmail.com>

Date: 2011-08-22 23:20:13

HTML generated by org-mode 6.33x in emacs 23

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值