opj1837 Balance(dp)

本文介绍了解决POJ 1837问题的一种动态规划方法,通过三维数组dp[i][j]记录挂了i个重物时平衡度为j的数目。文章详细解释了核心循环的逻辑,包括重物编号、背包和位置的循环顺序,并给出了完整的C语言代码实现。

http://poj.org/problem?id=1837

(1)用dp[i][j]表示挂了 i 个重物时 平衡度为 j 的数目;

(2)核心部分: 

for(j=1;j<=m;j++)
    for(k=0;k<=15000;k++)
        if(dp[j-1][k])
            for(i=1;i<=n;i++)
                dp[j][k+c[i]*g[j]]+=dp[j-1][k];

其中, j  表示重物编号, k  是背包, i  为所挂位置。注意循环的顺序,先确定重物,再打开背包,最后逐个转移平衡度。

(3)由于力矩的变化才是本题的关键,将中间位置设为平衡位置(dp[0][7500]=1;),力矩为7500时即为平衡。

 具体代码:

View Code
#include<stdio.h>
#include<string.h>
int n, m;
int g[21], c[21];
int dp[21][15001];
int main()
{
    int i, j, k;
    while(scanf("%d%d", &n, &m)!=EOF)
    {
        for(i=1;i<=n;i++) scanf("%d", &c[i]);
        for(j=1;j<=m;j++) scanf("%d", &g[j]);
        memset(dp, 0, sizeof(dp));
        dp[0][7500]=1;
        for(j=1;j<=m;j++)
            for(k=0;k<=15000;k++)
                if(dp[j-1][k])
                for(i=1;i<=n;i++)
                    dp[j][k+c[i]*g[j]]+=dp[j-1][k];
        printf("%d\n", dp[m][7500]);
    }
}

 

 

转载于:https://www.cnblogs.com/tim11/archive/2012/08/16/2642071.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值