http://acm.timus.ru/problem.aspx?space=1&num=1244
背包题 理解并不难
主要在于如果答案有多个要输出 -1
一个答案的话要输出结果 否则输出 0
用 sum [ n ] 表示到 n 有几条路径
状态转移方程为
if(sum[j-a[i]])//a[i]表示第i个数据的大小
{
        sum[j]=max(sum[j]+1,sum[j-a[i]]);
}
代码及其注释:
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define LL long long
using namespace std;
const int N=100003;
int sum[N];//到 N 有几条路径
int f[N];// 记录第一次到此用了那个数据
int a[103];// 数据大小
bool select[103];//是否用了
void Fselect(int x)//记录用了哪几个数据
{
    while(x)
    {
        select[f[x]]=true;
        x=x-a[f[x]];
    }
}
int main()
{
    //freopen("data.txt","r",stdin);
    int K,n;
    while(scanf("%d %d",&K,&n)!=EOF)
    {
        memset(sum,0,sizeof(sum));
        sum[0]=1;
        memset(select,false,sizeof(select));
        int m=0;
        for(int i=1;i<=n;++i)
        {
            scanf("%d",&a[i]);
            if(sum[K]>1)//如果答案有多个 则无需再找
            continue;
            m=min(m+a[i],K);//最大
            for(int j=m;j>=a[i];--j)
            {
                if(sum[j]==0)//如果是求得第一条到此路径 记录用了哪一数据
                {
                    f[j]=i;
                }
                if(sum[j-a[i]])//更新到此路径数目
                {
                   sum[j]=max(sum[j]+1,sum[j-a[i]]);
                }
            }
            if(sum[K]==1)
            Fselect(K);
        }
        if(sum[K]==0)
        {printf("0\n");continue;}
        if(sum[K]>1)
        {printf("-1\n");continue;}
        for(int i=1;i<=n;++i)
        {
            if(!select[i])
            printf("%d ",i);
        }
        printf("\n");
    }
    return 0;
}
                    
                
                
            
        
浙公网安备 33010602011771号