【JZOJ 1517】背包问题(dp)

问题描述
从T组物品中选出一些物品,放入背包中,求剩余空间的最小值。
限制条件:从每组物品中挑选物品必须要选取连续的一段。就是说,如果这组物品共有n个: 物品1、物品2、物品3、…、物品n,那么只能选取物品i、物品i+1、…、物品j,其中1<=i<=j<=n,或者不选。
输入
第一行为两个用空格隔开的正整数v和T。表示背包的空间和物品的组数。接下来有T行,每行先是一个正整数ni,表示这组物品有ni个,然后ni个正整数,表示每个物品的大小。
输出
仅一个数,表示剩余空间的最小值。
样例输入
100 3
3 7 6 8
2 80 70
4 101 108 103 150
样例输出
6
算法讨论
我们设b[i][j]为取第i种物品,是否存在所占空间为j的情况。因为要取连续的一段,我们可以处理一个前缀和,通过暴力枚举求出b数组。
设f[i][j]为取前i种物品,所占空间为j时能取得物品的最大占空间数。
则f[i][j]=max(f[i-1][j],f[i-1][j-k]+k),k满足存在b[i][k]且0<=k<=j。

#include 
using namespace std;
int a[16][106],s[16][106],f[16][5006],c[16];
bool b[16][5006];
int v,t,maxx;

int main()
{
    scanf("%d%d",&v,&t);
    for (int i=1;i<=t;i++)
    {
        scanf("%d",&c[i]);
        for (int j=1;j<=c[i];j++)
        {
            scanf("%d",&a[i][j]);
            s[i][j]=s[i][j-1]+a[i][j];
        }
    }
    for (int i=1;i<=t;i++)
        for (int j=1;j<=c[i];j++)
            for (int k=1;k<=j;k++)
                for (int l=1;l<=k;l++)
                    if (s[i][k]-s[i][l-1]<=v)
                        b[i][s[i][k]-s[i][l-1]]=1;
    for (int i=1;i<=t;i++)
        for (int j=0;j<=v;j++)
        {
            f[i][j]=f[i-1][j];
            for (int k=0;k<=j;k++)
                if (b[i][k] && f[i-1][j-k]+k>f[i][j])
                    f[i][j]=f[i-1][j-k]+k;
            if (f[i][j]>maxx)
                maxx=f[i][j];
        }
    printf("%d",v-maxx);
}

你可能感兴趣的:(动态规划,2018寒假中山纪中)