首页 > 其他 > 详细

洛谷P4095新背包问题

时间:2019-09-14 16:44:09      阅读:92      评论:0      收藏:0      [点我收藏+]

传送

技术分享图片
技术分享图片
技术分享图片
技术分享图片

这道题最最暴力的方法就是对于每一个询问都跑一边多重背包问题,但显然q不会那么友好的让我们用暴力过掉这道题。

考虑优化。我们可以先把裸的多重背包搞成二进制优化后的多重背包。但是复杂度依然无法接受。接下来使用吸氧和register等玄学优化 然而你发现你还是T了

那我们可不可以记录下来第i种不选,总容量为j($1\leq j\leq 1000$时的最大价值?想法很好,但是暴力写出来复杂度还是太高(O(\(1000n^2logn\)))

暴力写出来的

for(int i=1;i<=n;i++)//枚举去掉那一种
{
  for(int k=1;k<=t;k++)//t是二进制拆分后的物品总个数
  {
   if(k>=st[i]&&k<=en[i])continue;//st[i]为第i种物品在拆分后的第一个物品的编号,en[i]为第i种最后一个物品的编号
   for(int j=1000;j>=w[k];j--)
    f[i][j]=max(f[i][j],f[i][j-w[k]]+v[k]);
  }
}
    

上面的程序复杂度主要高在什么地方呢?f[i][j]和f[i-1][j]相比,考虑的物品多了第i-1种物品,少了第i种物品,而其他不变。但是上面的程序枚举哪一种物品不选后就全部重新考虑了一遍,会造成很大的浪费。

为了减少浪费,我们可以把上面的f[i][j]拆成两部分。可以先算出选1~i-1种物品最大价值,再算出选i+1~n种物品的最大价值,枚举合并的价值即可。

Code:
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<queue>
#include<cstring>
#include<cmath>
using namespace std;
const int inf=214748364;
typedef long long ll;
inline int read()
{
    char ch=getchar();
    int x=0;bool f=0;
    while(ch<'0'||ch>'9')
    {
        if(ch=='-')f=1;
        ch=getchar();
    }
    while(ch>='0'&&ch<='9')
    {
        x=(x<<3)+(x<<1)+(ch^48);
        ch=getchar();
    }
    return f?-x:x;
}
int n,w[10009],v[10009],q,t;
int st[1009],en[1009],me,meyo[300009],m[300009],dp[2009][1009],f[1009][1009],g[1009][1009];
bool have[1009][1009],ha[1009];
void Dp()
{
    for(register int duliu=1;duliu<=n;duliu++)
    {
        for(int j=1;j<=1000;j++)
         f[duliu][j]=f[duliu-1][j];
        for(int i=st[duliu-1];i<=en[duliu-1];i++)
         for(int j=1000;j>=w[i];j--)
          f[duliu][j]=max(f[duliu][j],f[duliu][j-w[i]]+v[i]); 
    }
    for(int duliu=n;duliu>=1;duliu--)
    {
        for(int j=1;j<=1000;j++)
         g[duliu][j]=g[duliu+1][j];
        for(int i=st[duliu+1];i<=en[duliu+1];i++)
         for(int j=1000;j>=w[i];j--)
          g[duliu][j]=max(g[duliu][j],g[duliu][j-w[i]]+v[i]); 
    }
}
int main()
{
    memset(st,0x3f,sizeof(st));
    n=read();
    for(register int i=1;i<=n;i++)
     {
        int mo=read(),va=read(),num=read();
        int k=1;
        st[i]=t+1;
        while(num>=k)
        {
            w[++t]=mo*k;
            v[t]=va*k;
            num-=k;
            k*=2;
        }
        if(num)
        {
            w[++t]=mo*num;
            v[t]=va*num;
        }
        en[i]=t;
     }
    q=read();
    for(register int i=1;i<=q;i++)
      meyo[i]=read()+1,m[i]=read(),have[meyo[i]][m[i]]=1,ha[meyo[i]]=1;
    Dp();
    for(register int i=1;i<=q;i++)
    {
        int ans=0;
        for(int j=0;j<=m[i];j++)
         ans=max(ans,f[meyo[i]][j]+g[meyo[i]][m[i]-j]);
        printf("%d\n",ans); 
    }  
}

然鹅这题正解是cdq分治,but我不会

洛谷P4095新背包问题

原文:https://www.cnblogs.com/lcez56jsy/p/11519298.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!