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\n",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号物品。

全部代码:

 

posted @ 2019-08-15 16:02  晔子  阅读(314)  评论(0编辑  收藏  举报