洛谷P4095新背包问题




这道题最最暴力的方法就是对于每一个询问都跑一边多重背包问题,但显然q不会那么友好的让我们用暴力过掉这道题。
考虑优化。我们可以先把裸的多重背包搞成二进制优化后的多重背包。但是复杂度依然无法接受。接下来使用吸氧和register等玄学优化 然而你发现你还是T了
那我们可不可以记录下来第i种不选,总容量为j($1\leq j\leq 1000$时的最大价值?想法很好,但是暴力写出来复杂度还是太高(O(\(1000n^2logn\)))
for(int i=1;i=st[i]&&k=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]拆成两部分。可以先算出选1i-1种物品最大价值,再算出选i+1n种物品的最大价值,枚举合并的价值即可。
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我不会
maybe窝搞完单调队列优化之后会回来写cdq分治的
希望上面那条不要变成最小鸽
洛谷P4095新背包问题的更多相关文章
- 洛谷P1242 新汉诺塔(dfs,模拟退火)
洛谷P1242 新汉诺塔 最开始的思路是贪心地将盘子从大到小依次从初始位置移动到目标位置. 方法和基本的汉诺塔问题的方法一样,对于盘子 \(i\) ,将盘子 \(1\to i-1\) 放置到中间柱子上 ...
- 洛谷P4095||bzoj3163 [HEOI2013]Eden 的新背包问题
https://www.luogu.org/problemnew/show/P4095 不太会.. 网上有神奇的做法: 第一种其实是暴力(复杂度3e8...)然而可以A.考虑多重背包,发现没有办法快速 ...
- 洛谷P1860 新魔法药水
洛谷题目链接 动态规划: 这个题目调了我好久....结果循环变量写错了... 而且题目有个坑!!!只能用开始给你的$v$元买入东西 回归正题: 我们定义状态$ans[i][j]$表示第$i$个物品用了 ...
- 题解——洛谷P4095 [HEOI2013]Eden 的新背包问题(背包)
思路很妙的背包 用了一些前缀和的思想 去掉了一个物品,我们可以从前i-1个和后i+1个推出答案 奇妙的思路 #include <cstdio> #include <algorithm ...
- 洛谷 P1305 新二叉树
P1305 新二叉树 题目描述 输入一串完全二叉树,用遍历前序打出. 输入输出格式 输入格式: 第一行为二叉树的节点数n. 后面n行,每一个字母为节点,后两个字母分别为其左右儿子. 空节点用*表示 输 ...
- 洛谷P1860——新魔法药水
传送门:QAQQAQ 题意:商店里有N种药水,每种药水都有一个售价和回收价.小S攒了V元钱,还会M种魔法,可以把一些药水合成另一种药水.他一天可以使用K次魔法,问他一天最多赚多少钱? N<=60 ...
- 洛谷 P1305 新二叉树 Label:字符串的输出总是有惊喜
题目描述 输入一串完全二叉树,用遍历前序打出. 输入输出格式 输入格式: 第一行为二叉树的节点数n. 后面n行,每一个字母为节点,后两个字母分别为其左右儿子. 空节点用*表示 输出格式: 前序排列的完 ...
- 洛谷P1242 新汉诺塔
传送门啦 首先要将第n个盘子从x到y,那么就要把比n小的盘子全部移到6-x-y,然后将n移到y 仔细想想:6代表的是3根初始柱,3根目标柱. 6-(x+y) 便是我们的中转柱了,因为到这个位置是最优的 ...
- 洛谷P1242 新汉诺塔 【神奇的递归】
题目描述 设有n个大小不等的中空圆盘,按从小到大的顺序从1到n编号.将这n个圆盘任意的迭套在三根立柱上,立柱的编号分别为A.B.C,这个状态称为初始状态. 现在要求找到一种步数最少的移动方案,使得从初 ...
随机推荐
- redis为什么使用单线程 ,还那么快,单线程是怎么实现的
单线程使用队列 为什么使用单线程 https://baijiahao.baidu.com/s?id=1628498089535886382&wfr=spider&for=pc http ...
- luogu P4631 [APIO2018] Circle selection 选圆圈
传送门 那个当前半径最大的圆可以用堆维护.这道题一个想法就是优化找和当前圆有交的圆的过程.考虑对于所有圆心建KD-tree,然后在树上遍历的找这样的点.只要某个点子树内的点构成的矩形区域到当前圆心的最 ...
- TS学习
随着vue3.0的即将到来,是时候学习一下TS了 简介:TypeScript是一种由微软开发的自由和开源的编程语言.它是JavaScript的一个超集,而且本质上向这个语言添加了可选的静态类型和基于类 ...
- 机器学习-非线性回归(Logistic Regression)及应用
1. 概率 1.1 定义:概率(Probability):对一件事情发生的可能性的衡量. 1.2 范围:0 <= P <= 1 1.3 计算方法: 1.3.1 根据个人置信 1.3.2 根 ...
- centos7安装部署jumpserver
一.系统环境准备1.查看系统版本 # cat /etc/redhat-release // 查看系统版本 CentOS Linux release (Core) # uname -a // 查看系统信 ...
- 修改docker自定义网桥后的默认网卡名称
[root@docker2 ~]# docker network create --subnet=10.10.10.0/24 docker1 #新键网桥docker142323044a4 ...
- Stanford CS229 Machine Learning by Andrew Ng
CS229 Machine Learning Stanford Course by Andrew Ng Course material, problem set Matlab code written ...
- JSONP 跨域请求原理
0x00 简介 由于浏览器的同源策略,我们想要从别的域获取数据变得困难,需要特殊的技术才能获取 0x01 使用 客户域:client.com 服务器(他域):server.com 如客户想访问 : h ...
- 数据库基本概念及Oracle基本语句
一.数据库分类 通常是按照数据模型的特点将传统数据库系统分成网状数据库.层次数据库和关系数据库三种. 1.网状数据库 顾名思义,网状数据库采用的是以记录类型为结点的网状数据模型,是一种导航式(Navi ...
- 区分Overloading、Overriding及Hiding
在面向对象(OO)的世界中存在着三个十分容易混淆的概念:重载(Overloading).重写(Overriding).隐藏(Hiding). 1.重载 重载是指同一作用域的不同函数使用相同的函数名,但 ...