CodeForces 687C【DP】

题意:
给你n个数,然后让这些数相加组合,然后在这些组合的数里可以再相加组合搞出给定 k,输出这些组合的数。

思路:
DP。
//在枚举到第i个coin的时,dp[i][j],i 肯定能被a[i]组合,
//然后再枚举<=a[i]的部分,dp[i][j]的具体意义就是在coin值是i的时候,能用j去组合。
//为了防止重复利用coin,从j枚举到a[i];
//最后dp[k][h]==1的把h塞到容器里去,最后输出。

#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
typedef unsigned long long ULL;
const double eps=1e-6;
const double pi=acos(-1.0);
const int mod=998244353;
const int INF=0x3f3f3f3f;


const int N=2550;

int a[510];
vector<int>d;
bool dp[N][N];
int k,n;

int main()
{
    int i,j,h;
    cin>>n>>k;
    for(i=0;i<n;i++){
        scanf("%d",&a[i]);
    }
    memset(dp,0,sizeof(dp));
    dp[0][0]=1;
    for(i=0;i<n;i++){
        for(j=k;j>=a[i];j--){
            for(h=0;h<=k-a[i];h++){
                if(dp[j-a[i]][h])
                    dp[j][h]=dp[j][h+a[i]]=1;
            }
        }
    }

    for(i=0;i<=k;i++){
        if(dp[k][i])
            d.push_back(i);
    }
    sort(d.begin(),d.end());
    int len=d.size();
    printf("%d\n",len);
    for(i=0;i<len;i++){
        if(i) printf(" ");
        printf("%d",d[i]);
    }
    puts("");
    return 0;
}
posted @ 2016-07-18 16:33  see_you_later  阅读(124)  评论(0编辑  收藏  举报