男人八题中的签到题,因为懒得用单调队列优化,所以用了二进制优化,但是这样很卡时限

我第一次是用int数组来表示dp数组,然后通过01背包的加法求,之后判断,但是这样memset的时间会很长,所以建议用bool数组+或判断,因为我们只在乎是否存在而不在乎多少

POJ1742 Coins(多重背包+二进制优化)_数组POJ1742 Coins(多重背包+二进制优化)_动态规划_02
#include<iostream>
#include<queue>
#include<map>
#include<vector>
#include<cstdio>
#include<algorithm>
#include<stack>
#include<cstring>
using namespace std;
typedef long long ll;
const int N=1e5+7;
const int inf=0x3f3f3f3f;
int a[110],b[110];
int n,m;
int num[N];
bool f[N];
int main(){
    while(cin>>n>>m){
    if(n==0||m==0)
    break;
    int i;
    for(i=1;i<=n;i++)
        scanf("%d",&a[i]);
    for(i=1;i<=n;i++)
        scanf("%d",&b[i]);
    int cnt=0;
    for(i=1;i<=n;i++){
        int k=1;
        while(k<=b[i]){
            num[++cnt]=a[i]*k;
            b[i]-=k;
            k<<=1;
        }
        if(b[i]){
            num[++cnt]=a[i]*b[i];
        }
    }
    int j;
    memset(f,0,sizeof f);
    f[0]=1;
    for(i=1;i<=cnt;i++){
        for(j=m;j>=num[i];j--){
            f[j]|=f[j-num[i]];
        }
    }
    int res=0;
    for(i=1;i<=m;i++){
        if(f[i])
            res++;
    }
    printf("%d\n",res);
    }

}
View Code

 

没有人不辛苦,只有人不喊疼