hdu2844 Coins(多重背包)

母函数TLE...

//法1
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

 

#define MAXN 105
#define MAXM 100005

int sum, value[MAXN], num[MAXN];
int f[MAXM];

inline int getMax(int a, int b)
{
    return a > b ? a : b;
}

void ZeroOnePack(int w, int v)
{
    for(int j = sum; j >= w; j--)
    {
        f[j] = getMax(f[j], f[j - w] + v);
    }
}
void CompletePack(int w, int v)
{
    for(int j = w; j <= sum; j++)
    {
        f[j] = getMax(f[j], f[j - w] + v);
    }
}
void MultiplePack(int w, int v, int amount)
{
    if(amount * w >= sum)
    {
        CompletePack(w, v);
        return;
    }
    int k = 1;
    while(k < amount)
    {
        ZeroOnePack(k * w, k * v);
        amount -= k;
        k <<= 1;
    }
    ZeroOnePack(amount * w, amount * v);
}

int main()
{
#ifndef ONLINE_JUDGE
    freopen("indata.txt", "r", stdin);
#endif
    int n;
    while(scanf("%d %d", &n, &sum), n + sum)
    {
        int i;
        for(i = 1; i <= n; i++) scanf("%d", &value[i]);
        for(i = 1; i <= n; i++) scanf("%d", &num[i]);

        memset(f, 0, sizeof(f));
        for(i = 1; i <= n; i++) MultiplePack(value[i], value[i], num[i]);
        int cnt = 0;
        for(i = 1; i <= sum; i++)
        {
        //    printf("%d ", f[i]);
            if(i == f[i]) cnt++;
        }
        printf("%d\n", cnt);
    }
    return 0;
}

//法2
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define MAXN 105
#define MAXM 100005

int sum, value[MAXN], num[MAXN];
bool f[MAXM];

void ZeroOnePack(int w, int v)
{
    for(int j = sum; j >= w; j--)
    {
        f[j] = f[j] || f[j - w];
    }
}
void CompletePack(int w, int v)
{
    for(int j = w; j <= sum; j++)
    {
        f[j] = f[j] || f[j - w];
    }
}
void MultiplePack(int w, int v, int amount)
{
    if(amount * w >= sum)
    {
        CompletePack(w, v);
        return;
    }
    int k = 1;
    while(k < amount)
    {
        ZeroOnePack(k * w, k * v);
        amount -= k;
        k <<= 1;
    }
    ZeroOnePack(amount * w, amount * v);
}

int main()
{
#ifndef ONLINE_JUDGE
    freopen("indata.txt", "r", stdin);
#endif
    int n;
    while(scanf("%d %d", &n, &sum), n + sum)
    {
        int i;
        for(i = 1; i <= n; i++) scanf("%d", &value[i]);
        for(i = 1; i <= n; i++) scanf("%d", &num[i]);

        memset(f, 0, sizeof(f));
        f[0] = 1;
        for(i = 1; i <= n; i++) MultiplePack(value[i], value[i], num[i]);
        int cnt = 0;
        for(i = 1; i <= sum; i++)
        {
        //    printf("%d ", f[i]);
            if(f[i]) cnt++;
        }
        printf("%d\n", cnt);
    }
    return 0;
}

posted @ 2010-10-19 01:10  菜到不得鸟  阅读(129)  评论(0)    收藏  举报