zoukankan      html  css  js  c++  java
  • CODEVS 3269 混合背包

    一道裸的混合背包题目。可是忘记了去重一直TLE,就是假设体积<=全然背包的01。和多重背包都要被全然背包代替,由于他的数量没限制所以用起来更方便。

    题目连接:http://codevs.cn/problem/3269/

    #include <iostream>
    #include <string.h>
    #include <cstdio>
    #include <cstdlib>
    using namespace std;
    
    const int maxn = 210;
    const int maxc = 200010;
    
    
    inline int read()
    {
        char ch;
        bool flag = false;
        int a = 0;
        while(!((((ch = getchar()) >= '0') && (ch <= '9')) || (ch == '-')));
        if(ch != '-')
        {
            a *= 10;
            a += ch - '0';
        }
        else
        {
            flag = true;
        }
        while(((ch = getchar()) >= '0') && (ch <= '9'))
        {
            a *= 10;
            a += ch - '0';
        }
        if(flag)
        {
            a = -a;
        }
        return a;
    }
    void write(int a)
    {
        if(a < 0)
        {
            putchar('-');
            a = -a;
        }
        if(a >= 10)
        {
            write(a / 10);
        }
        putchar(a % 10 + '0');
    }
    
    int f[maxc], v[maxn], w[maxn], m[maxn];
    int n, V;
    bool vis[maxn];
    
    void complete_pack(int cost, int weight)
    {
        for(int j = cost; j <= V; j++)
            f[j] = max(f[j], f[j-cost]+weight);
    }
    
    void zero_pack(int cost, int weight)
    {
        for(int j = V; j >= cost; j--)
            f[j] = max(f[j], f[j-cost]+weight);
    }
    
    void multi_pack(int cost, int weight, int amount)
    {
        if(cost*amount >= V)
        {
            complete_pack(cost, weight);
            return;
        }
        int k = 1;
        while(k < amount)
        {
            zero_pack(cost*k, weight*k);
            amount -= k;
            k <<= 1;
        }
        zero_pack(amount*cost, amount*weight);
    }
    int main()
    {
        n = read();
        V = read();
        memset(vis, false, sizeof(vis));
        for(int i=1; i<=n; i++)
        {
            v[i] = read();
            w[i] = read();
            m[i] = read();
        }
        for(int i = 1; i <= n; i++)
        {
            for(int j = i+1; j <= n; j++)
            {
                if(m[i] == 1)
                {
                    if(m[j] == -1)if(v[i] >= v[j] && w[i] <= w[j]) vis[i] = true;
                    if(m[j] > 1)if(v[i] >= v[j] && w[i] <= w[j]) vis[i] = true;
                }
                else if(m[i] > 1)
                {
                    if(m[j] == -1)
                    if(v[i] >= v[j] && w[i] <= w[j]) vis[i] = true;
                }
            }
        }
        for(int i=1; i<=n; i++)
        {
            if(m[i]==1 && !vis[i])
                zero_pack(v[i], w[i]);
            else if(m[i]==-1)
                complete_pack(v[i], w[i]);
            else if(!vis[i])multi_pack(v[i], w[i],m[i]);
        }
        write(f[V]);
        return 0;
    }
    


  • 相关阅读:
    地图 SDK 系列教程-在地图上展示指定区域
    [奇思妙想]下一个和微博、微信同级别的应用为是什么样的
    [办公自动化]EXCEL不大,但是保存很慢
    [奇思妙想]公共图书馆+快递
    [奇思妙想]“停哪了”
    [IT学习]阿铭Linux 微信公众号 每日一题 解析
    [IT学习]GIT 学习
    [故障处理]西部数据wd elements xp 无法识别
    [奇思妙想]无人机
    [IT学习]跟阿铭学linux(第3版)
  • 原文地址:https://www.cnblogs.com/clnchanpin/p/6897928.html
Copyright © 2011-2022 走看看