Luogu P4095 [HEOI2013]Eden的新背包问题
题目
求出从前往后的背包\(f_{i,j}\)和从后往前的背包\(F_{i,j}\)。
那么对于询问\((d,e)\),答案就是\(\max\limits_{i=0}^e f_{d-1,i}+F_{d+1,e-i}\)。
然后就是单调队列优化多重背包。
记物品有\(c[i]\)个,价值为\(v[i]\),代价为\(w[i]\)。
多重背包的转移\(f[i][j]=\max\limits_{k=0}^{min(c[i],\lfloor\frac j{w[i]}\rfloor)}(f[i-1][j-w[i]*k]+v[i]*k)\)
令\(s=\lfloor\frac j{w[i]}\rfloor,d=j-w[i]*s\)。
则\(f[i][j]=\max\limits_{k=0}^{min(c[i],\lfloor\frac j{w[i]}\rfloor)}(f[i-1][d+(s-k)*w[i]]+v[i]*k)\)
令\(k=s-k\),则\(f[i][j]=\max\limits_{k=max(0,s-c[i])}^{s}(f[i-1][d+k*w[i]]-v[i]*k)+v[i]*s\)
也就是对于\(f[i][d+k*w[i]\),我们需要找到前面\(f[i-1][d+K*w[i]]-K*w[i](K\in[k-c[i],k])\)的最大值,然后加上\(k*w[i]\)。
我们将前面的所有\(f[i-1][d+K*w[i]]-K*w[i](K\in[k-c[i],k])\)放进一个单调队列,每次把队首的\(K\)小于\(k-c[i]\)的弹出,然后把当前的\(f[i][d+k*w[i]]\)加入队尾,然后取出队首更新答案。
这里我们可以用一个pair的deque来实现。
#include<bits/stdc++.h>
#define mp make_pair
#define P pair<int,int>
#define fir first
#define sec second
using namespace std;
const int N=1007,V=10007;
int read(){int x;scanf("%d",&x);return x;}
void max(int &a,int b){a=a>b? a:b;}
int c[N],w[N],v[N],d[N],e[N],f[N][V],F[N][V];
void cal(int *f,int i)
{
for(int d=0,k;d<w[i];++d)
{
deque<P>q{mp(0,f[d])};
for(k=1;k*w[i]+d<=10000;++k)
{
while(!q.empty()&&q.front().fir<k-c[i]) q.pop_front();
while(!q.empty()&&q.back().sec<=f[k*w[i]+d]-k*v[i]) q.pop_back();
q.push_back(mp(k,f[k*w[i]+d]-k*v[i])),max(f[k*w[i]+d],q.front().sec+k*v[i]);
}
}
}
int main()
{
int i,d,e,n,q,ans;
for(n=read(),i=1;i<=n;++i) w[i]=read(),v[i]=read(),c[i]=read();
for(i=1;i<=n;++i) memcpy(f[i],f[i-1],sizeof f[i]),cal(f[i],i);
for(i=n;i;--i) memcpy(F[i],F[i+1],sizeof F[i]),cal(F[i],i);
for(q=read();q;--q)
{
d=read()+1,e=read(),ans=0;
for(i=0;i<=e;++i) max(ans,f[d-1][i]+F[d+1][e-i]);
printf("%d\n",ans);
}
}
或者换成手写队列也行。不过我刚刚写锅了,懒得写了。反正deque常数挺小的。
Luogu P4095 [HEOI2013]Eden的新背包问题的更多相关文章
- luogu P4095 [HEOI2013]Eden 的新背包问题 多重背包 背包的合并
LINK:Eden 的新背包问题 就是一个多重背包 每次去掉一个物品 询问钱数为w所能买到的最大值. 可以对于每次Q暴力dp 利用单调队列优化多重背包 这样复杂度是Qnm的. 发现过不了n==10的点 ...
- LUOGU P4095 [HEOI2013]Eden 的新背包问题
题目描述 " 寄 没 有 地 址 的 信 ,这 样 的 情 绪 有 种 距 离 ,你 放 着 谁 的 歌 曲 ,是 怎 样 的 心 情 . 能 不 能 说 给 我 听 ." 失忆的 ...
- Luogu P4095 [HEOI2013]Eden 的新背包问题 思维/动规
当时一直在想前缀和...多亏张队提醒... 从1到n背次包,保存每一个状态下的价值,就是不要把第一维压掉:再从n到1背一次,同样记住每种状态: 然后询问时相当于是max(前缀+后缀),当然前缀后缀中间 ...
- P4095 [HEOI2013]Eden 的新背包问题
P4095 [HEOI2013]Eden 的新背包问题 题解 既然假定第 i 个物品不可以选,那么我们就设置两个数组 dpl[][] 正序选前i个物品,dpr[][] 倒序选前i个物品 ,价格不超过 ...
- 题解——洛谷P4095 [HEOI2013]Eden 的新背包问题(背包)
思路很妙的背包 用了一些前缀和的思想 去掉了一个物品,我们可以从前i-1个和后i+1个推出答案 奇妙的思路 #include <cstdio> #include <algorithm ...
- BZOJ 3163: [Heoi2013]Eden的新背包问题( 背包dp )
从左到右, 从右到左分别dp一次, 然后就可以回答询问了. ---------------------------------------------------------- #include< ...
- BZOJ3163&Codevs1886: [Heoi2013]Eden的新背包问题[分治优化dp]
3163: [Heoi2013]Eden的新背包问题 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 428 Solved: 277[Submit][ ...
- 洛谷P4095||bzoj3163 [HEOI2013]Eden 的新背包问题
https://www.luogu.org/problemnew/show/P4095 不太会.. 网上有神奇的做法: 第一种其实是暴力(复杂度3e8...)然而可以A.考虑多重背包,发现没有办法快速 ...
- bzoj 3163: [Heoi2013]Eden的新背包问题
Description "寄没有地址的信,这样的情绪有种距离,你放着谁的歌曲,是怎样的心心静,能不能说给我听."失忆的Eden总想努力地回忆起过去,然而总是只能清晰地记得那种思念的 ...
随机推荐
- Nowcoder Playing Games ( FWT 优化 DP && 博弈论 && 线性基)
题目链接 题意 : 给出 N 个数.然后问你最多取出多少石子使得在 NIM 博弈中.后手必胜 分析 : Nim 博弈模型,后手必胜当且仅当各个堆的石子的数目的异或和为 0 转化一下.变成最少取多少石 ...
- UVa 1595 Symmetry (set && math)
题意:给出n个在直角坐标系上的点,问你能不能找出一条竖轴(即垂直于x的轴)使得所有的点根据这条轴对称,能则输出YES,否则输出NO 分析:首先需要找到对称轴的值,将所有n个点的x轴的值加起来然后去除以 ...
- Spark译文(二)
PySpark Usage Guide for Pandas with Apache Arrow(使用Apache Arrow的Pandas PySpark使用指南) Apache Arrow in ...
- 高效的js原生代码
1.遍历元素 //不推荐 var element = document.getElementsByTagName('div'); for(var i=0; i<element.length; i ...
- Centos 6.5 Multipath 初始配置
# This is a basic configuration file with some examples, for device mapper # multipath. # For a comp ...
- UVA 122 -- Trees on the level (二叉树 BFS)
Trees on the level UVA - 122 解题思路: 首先要解决读数据问题,根据题意,当输入为“()”时,结束该组数据读入,当没有字符串时,整个输入结束.因此可以专门编写一个rea ...
- 使用同步上下文进行C#与VBA代码和Excel之间的交互
原始出处:www.cnblogs.com/Charltsing/p/RunVBA.html 大家都知道,Excel是个STA,不允许在Excel忙的时候对其Com对象进行操作,也不允许同时有多个线程对 ...
- leecode 238除自身以外数组的乘积
class Solution { public: vector<int> productExceptSelf(vector<int>& nums) { //用除法必须要 ...
- django路由的二级分发
基于二级分发设计url路由 path('index/', views.index), path('index/', ([ path('test01/', test01), path('test02/' ...
- Prism学习--实现可插拔的模块
首先,在使用Prism框架加载的程序集中分别添加一个类,并让这些类实现IModule接口.当Prism框架加载某个程序集后,将首先在程序集中搜索实现了该接口的类.之后将会调用该接口的Initializ ...