zoukankan      html  css  js  c++  java
  • P4095 [HEOI2013]Eden 的新背包问题

    P4095 [HEOI2013]Eden 的新背包问题

    题解

    既然假定第 i 个物品不可以选,那么我们就设置两个数组

    dpl[][] 正序选前i个物品,dpr[][] 倒序选前i个物品 ,价格不超过 j 的最大价值

    然后正着反着跑 多重背包 

    最后答案考虑 i 之前的物品的价格和  i 之后的物品的价格,转移如下:

    代码

    #include<bits/stdc++.h>
    
    using namespace std;
    
    inline int read()
    {
        int ans=0;
        char last=' ',ch=getchar();
        while(ch<'0'||ch>'9') last=ch,ch=getchar();
        while(ch>='0'&&ch<='9') ans=ans*10+ch-'0',ch=getchar();
        if(last=='-') ans=-ans;
        return ans;
    }
    
    int n,q,d,e,ans=0;
    int dpl[1005][1005],dpr[1005][1005];
    //dpl[][]正序选前i个物品,dpr[][]倒序选前i个物品 
    struct node{
        int a,b,c;
    }thing[1005];
    
    void pre() //多重背包 
    {
        for(int i=1;i<=n;i++)  //正序背包
        {
            for(int j=0;j<=1000;j++) dpl[i][j]=dpl[i-1][j];
            int x=1,z=thing[i].c ;
            while(x<=z){
                for(int k=1000;k>=thing[i].a *x;k--)
                   dpl[i][k]=max(dpl[i][k],dpl[i][k-thing[i].a *x]+thing[i].b *x);
                z-=x;
                x<<=1;
            }if(z){
                for(int k=1000;k>=thing[i].a *z;k--)
                   dpl[i][k]=max(dpl[i][k],dpl[i][k-thing[i].a *z]+thing[i].b *z);
            }
        }
        for(int i=n;i>=1;i--) //倒序背包
        {
            for(int j=0;j<=1000;j++) dpr[i][j]=dpr[i+1][j];
            int x=1,z=thing[i].c ;
            while(x<=z){
                for(int k=1000;k>=thing[i].a *x;k--)
                   dpr[i][k]=max(dpr[i][k],dpr[i][k-thing[i].a *x]+thing[i].b *x);
                z-=x;
                x<<=1;
            }if(z){
                for(int k=1000;k>=thing[i].a *z;k--)
                   dpr[i][k]=max(dpr[i][k],dpr[i][k-thing[i].a *z]+thing[i].b *z);
            }
        }
    }
    
    int main()
    {
        n=read();
        for(int i=1;i<=n;i++)
          thing[i].a =read(),thing[i].b =read(),thing[i].c =read();
        pre();
        q=read();
        for(int i=1;i<=q;i++)
        {
            d=read()+1; //输入编号是从0开始的 
            e=read();ans=0;
            for(int j=0;j<=e;j++)
              ans=max(ans,dpl[d-1][j]+dpr[d+1][e-j]);
            printf("%d
    ",ans);
        }    
        return 0;
    }

    后面附上听课笔记:

    ◦ N个物品,第i个物品有c[i]个,购买第i个物品需要a[i]元,可获利b[i]的价
    值。有m个询问,每次询问:如果第x个物品禁止购买,你有y元的话,能
    获得的最大价值是多少?询问之间互相独立。
    ◦ N<=1000,m<=3*10^5 
    >Solution

    这是一个经典的问题
    ◦ 分治+背包
    ◦ 初始solve(1,n)
    ◦ 递归的函数到Solve(l,r),维护的dp数组,记录的是除去[l,r]外的物品的构成的背
    包数组。
    ◦ Solve(l,mid)时,把[mid+1,r]内的物品加入dp数组。
    ◦ 我们这里定义的加入这个物品u,就是多考虑上这个物品之后构成的dp数组。
    若是0/1背包的加入也就是做以下这个操作。
    ◦ For (int i=n;i>=w[u];i--) dp[i]=max(dp[i],dp[i-w[u]]+v[u]);
    ◦ l=r时,将对应所有的询问在dp数组查询即可。
    ◦ 单调队列优化的话,复杂度O(n*m*log(n)),每个物品被加进去log次,每次O(m) 

    代码实现

    。由于你要分治求解,所以d表示深度

    对于每次询问,ans得到答案,id[i]表示询问编号,S[i]表示题目所给出的y

    。由于是分治,所以每次都先复制一遍再传递下去

     

    ◦ Insert(dp,i):是在dp数组当中加入i号物品。

    全部代码:

  • 相关阅读:
    C#面向对象编程基础-喜课堂笔记
    [爬虫]通过url获取连接地址中的数据
    第10季asp.net基础
    初学MVC
    学习MVC遇到的问题
    飞行棋小项目
    JAVAscript学习笔记
    iOS 清除xcode缓存和生成文件
    Access用OleDbParameter更新/插入数据
    SQLite动态库下载
  • 原文地址:https://www.cnblogs.com/xiaoyezi-wink/p/11358527.html
Copyright © 2011-2022 走看看