使用一种二进制的优化, 可以完美的解决这题,《背包九讲》中说的非常好

但是还有一种线性复杂的算法。 应该算是该题很巧妙的解法

for(i=1;i<=6;i++)  
        {  
            for(l=total;l>=0;l--)  
            {  
                if(dp[l]==0)    continue;  
                for(k=1;k<=num[i]&&k*i+l<=total;k++)    
                {  
                    if(dp[k*i+l])   break;  // 这个剪枝瞬间将复杂度从N^2变成了N。 
                    dp[k*i+l]=1;  
                }  
            }  
        }  

代码中total是我们要装满的容量, 循环的次序很重要。 当关键还是那一步剪枝

 

在附上一份用二进制优化多重背包的代码

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <algorithm>
#include <math.h>
#include <map>
#include <queue>
#include <sstream>
#include <iostream>
using namespace std;
#define INF 0x3fffffff
struct node
{
    int w,c;
}g[10010]; //将log2压缩后的元素存在这里!

int k[10];
int dp[2002000];

int mpow(int x)
{
    int sum=1;
    for(int i=0;i<x;i++)
        sum=sum*2;
    return sum;
}

int main()
{
    //freopen("//home//chen//Desktop//ACM//in.text","r",stdin);
    //freopen("//home//chen//Desktop//ACM//out.text","w",stdout);
    int tt=1;
    while(1)
    {
        int tmp=0,sum=0;
        for(int i=1;i<=6;i++)
        {
            scanf("%d",&k[i]);
            tmp+=k[i];
            sum+=k[i]*i;
        }
        if(tmp==0) break;
        printf("Collection #%d:\n",tt++);
        if(sum%2!=0)
        {
            printf("Can't be divided.\n");
            printf("\n");
            continue;
        }
        int cnt=0;
        for(int i=1;i<=6;i++)
        {
            if(k[i]==0) continue;
            tmp=0;
            int tk=1;
            while(2*tk-1 < k[i])
            {
                g[cnt].w=1;
                g[cnt].c=i*tk;
                cnt++;
                tmp+=tk;
                tk*=2;
            }
            g[cnt].w=1;
            g[cnt].c=i*(k[i]-tmp);
            cnt++;
        }
        for(int i=1;i<=sum/2;i++)
        {
            dp[i]=-INF;
        }
        dp[0]=0;
        for(int i=0;i<cnt;i++)
        {
            for(int j=sum/2;j>=g[i].c;j--)
            {
                dp[j]=max(dp[j],dp[j-g[i].c]+g[i].w);
            }
        }
        if(dp[sum/2]>=0)
            printf("Can be divided.\n");
        else printf("Can't be divided.\n");
        printf("\n");
    }
    return 0;
}