VIJOS1392拼拼图的小杉

题目大意

给定n个数,将这n个数中的一些依次放进m个集合,每个集合中所有数的和不能超过T。集合包含的元素不能交叉,也就是说如果第1个数和第3个数放入了集合1,那么第2个数要么放入集合1,要么不放入任何一个集合。
求m个集合中最多能包含多少个数

1<=n,m,T<=1000

分析

来源于题解中小岛的想法

这题如果状态直接定义为f[i,j]=x表示前i个数放入j个集合最多能包含x个数
那么操作起来会比较困难,所有应换一种状态的定义。

记a[i]为第i个数,f[i,j]=(t,s)表示前i个数选了j个数需要t个集合,最后一个集合所有数之和为s,那么
f[i,j]=min{f[i-1,j],f[i-1,j-1]+a[i]}
f[0,0]=(1,0)
那么就是一个0-1背包问题啦
最后的答案是使得f[n,i]的t<=m的最大的i

代码

#include<cstdio>
const int maxn=1000;
int n,m,T,a[maxn+10];
struct node{
    int t,s;
    bool operator < (const node d) const{
        if (t<d.t)
            return 1;
        if (t>d.t)
            return 0;
        return s<d.s;
    }
    node operator + (const int d) const{
        if (s+d<=T)
            return (node){t,s+d};
        else
            return (node){t+1,d};
    }
};
node f[maxn+10];
int main(){
    scanf("%d%d%d",&n,&m,&T);
    node te;
    for (int i=1;i<=n;i++)
        scanf("%d",&a[i]);
    f[0].t=1;
    for (int i=1;i<=n;i++){
        f[i]=f[i-1]+a[i];
        for (int j=i-1;j>=1;j--){
            te=f[j-1]+a[i];
            if (te<f[j])
                f[j]=te;
        }
    }
    for (int i=n;i>=0;i--)
        if (f[i].t<=m){
            printf("%d\n",i);
            return 0;
        }
}

你可能感兴趣的:(动态规划,vijos)