题目地址:http://acm.hdu.edu.cn/showproblem.php?pid=3401

题目意思:

有T天,你每天可以以API买进,BPI卖出,最多买ASI个,最多卖BSI个

最多只能持有MAXP个商品,相邻两次交易要大于W天

问你最多在第T天结束的时候能赚多少钱

解题思路:

有三个状态转移方程,其中dp[i][j]表示第i天手上有j个商品时我赚的钱

dp[i][j] = max(dp[i][j],dp[i-1][j])   //即针对前一天我啥也不做

对于买而言,有一个递推式

dp[i][j] = max(dp[i][j],max(dp[i-w-1][k] - (j-k)*buyp[i]  )

进一步拆开就是 dp[i][j] = max(dp[i][j], max(dp[i-w-1][k]+k*buyp[i]) - j*buyp[i] );

因为是买,所以0<=(j-k)<=canbuyn[i]

我们可以利用优先队列将对于i来说能看到的最大放在队首

对于卖而言同理,只不过方程变成

dp[i][j] = max(dp[i][j], max(dp[i-w-1][k]+k*sellp[i])-j*sellp[i]

0<=(k-j)<=canselln[i]

下面上代码:

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<cmath>
#include<cstdlib>
using namespace std; const int maxn = 3000; struct node
{
int val,pos;
}q[maxn]; int sellp[maxn],buyp[maxn];
int canselln[maxn],canbuyn[maxn];
int dp[maxn][maxn];
int T,n,maxp,w;
const int inf = 0x3f3f3f3f; void dpf()
{
for(int i=1;i<=n;i++)
{
for(int j=0;j<=maxp;j++)
dp[i][j] = -inf;
} //在前w+1天不能卖,只能买
for(int i=1;i<=n;i++)
{
for(int j=0;j<=min(maxp,canbuyn[i]);j++)
{
dp[i][j] = -buyp[i]*j; //买的都是负账
}
} for(int i=2;i<=n;i++) //针对前一天我啥也不做
{
for(int j=0;j<=maxp;j++)
dp[i][j] = max(dp[i][j],dp[i-1][j]);
} for(int i=w+2;i<=n;i++) //前w+1天除了买啥也不能做,既能买又能卖得从w+2开始
{
int pre = i-w-1;
//对于买而言,有一个递推式
//dp[i][j] = max(dp[i][j],max(dp[i-w-1][k] - (j-k)*buyp[i] )
//进一步拆开就是 dp[i][j] = max(dp[i][j], max(dp[i-w-1][k]+k*buyp[i]) - j*buyp[i] );
//因为是买,所以0<=(j-k)<=canbuyn[i]
//我们可以利用优先队列将对于i来说能看到的最大放在队首
int head,tail;
head = 0;
tail = -1;
for(int j=0;j<=maxp;j++)
{
dp[i][j] = max(dp[i][j],dp[i-1][j]);
node tmp;
tmp.val = dp[pre][j]+j*buyp[i];
tmp.pos = j;
//插入到优先队列里面去
while(head<=tail && q[tail].val < tmp.val) tail--;
q[++tail] = tmp;
//队首还要满足0<=(j-k)<=canbuyn[i]
while(head<=tail && (j-q[head].pos)>canbuyn[i] )head++; //取队首
if(head<=tail) dp[i][j] = max(dp[i][j],q[head].val-j*buyp[i]);
} //对于卖而言同理,只不过方程变成
//dp[i][j] = max(dp[i][j], max(dp[i-w-1][k]+k*sellp[i])-j*sellp[i]
//0<=(k-j)<=canselln[i]
head=0;
tail=-1;
for(int j=maxp;j>=0;j--)
{
dp[i][j] = max(dp[i][j],dp[i-1][j]);
node tmp;
tmp.val = dp[pre][j]+j*sellp[i];
tmp.pos = j; while(head<=tail && q[tail].val < tmp.val) tail--;
q[++tail] = tmp; while(head<=tail && (q[head].pos-j) > canselln[i]) head++; if(head<=tail) dp[i][j] = max(dp[i][j],q[head].val-j*sellp[i]); } } } int main()
{
scanf("%d",&T);
while(T--)
{
scanf("%d%d%d",&n,&maxp,&w);
for(int i=1;i<=n;i++)
scanf("%d%d%d%d",&buyp[i],&sellp[i],&canbuyn[i],&canselln[i]);
dpf();
int ans = 0;
for(int j=0;j<=maxp;j++)
ans = max(ans,dp[n][j]);
printf("%d\n",ans);
}
return 0;
}

Trade-----HDU3401----单调队列优化的DP的更多相关文章

  1. 【NOIP2017】跳房子 题解(单调队列优化线性DP)

    前言:把鸽了1个月的博客补上 ----------------- 题目链接 题目大意:机器人的灵敏性为$d$.每次可以花费$g$个金币来改造机器人,那么机器人向右跳的范围为$[min(d-g,1),m ...

  2. 【HDU 3401 Trade】 单调队列优化dp

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3401 题目大意:现在要你去炒股,给你每天的开盘价值,每股买入价值为ap,卖出价值为bp,每天最多买as ...

  3. 【HDU】3401:Trade【单调队列优化DP】

    Trade Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submi ...

  4. 洛谷p1725 露琪诺 单调队列优化的DP

    #include <iostream> #include <cstdio> #include <cstring> using namespace std; int ...

  5. BestCoder Round #89 02单调队列优化dp

    1.BestCoder Round #89 2.总结:4个题,只能做A.B,全都靠hack上分.. 01  HDU 5944   水 1.题意:一个字符串,求有多少组字符y,r,x的下标能组成等比数列 ...

  6. 单调队列以及单调队列优化DP

    单调队列定义: 其实单调队列就是一种队列内的元素有单调性的队列,因为其单调性所以经常会被用来维护区间最值或者降低DP的维数已达到降维来减少空间及时间的目的. 单调队列的一般应用: 1.维护区间最值 2 ...

  7. bzoj1855: [Scoi2010]股票交易 单调队列优化dp ||HDU 3401

    这道题就是典型的单调队列优化dp了 很明显状态转移的方式有三种 1.前一天不买不卖: dp[i][j]=max(dp[i-1][j],dp[i][j]) 2.前i-W-1天买进一些股: dp[i][j ...

  8. hdu3401:单调队列优化dp

    第一个单调队列优化dp 写了半天,最后初始化搞错了还一直wa.. 题目大意: 炒股,总共 t 天,每天可以买入na[i]股,卖出nb[i]股,价钱分别为pa[i]和pb[i],最大同时拥有p股 且一次 ...

  9. 【单调队列优化dp】HDU 3401 Trade

    http://acm.hdu.edu.cn/showproblem.php?pid=3401 [题意] 知道之后n天的股票买卖价格(api,bpi),以及每天股票买卖数量上限(asi,bsi),问他最 ...

  10. 单调队列优化DP,多重背包

    单调队列优化DP:http://www.cnblogs.com/ka200812/archive/2012/07/11/2585950.html 单调队列优化多重背包:http://blog.csdn ...

随机推荐

  1. android KK版本号,如何更改蓝牙设备类型

    mediatek/external/bluetooth/bt_cust/bt_cust_table.h   {         .name = "ClassOfDevice",   ...

  2. UiAutomator源码分析之获取控件信息

    根据上一篇文章<UiAutomator源码分析之注入事件>开始时提到的计划,这一篇文章我们要分析的是第二点: 如何获取控件信息 我们在测试脚本中初始化一个UiObject的时候通常是像以下 ...

  3. [译]Java 设计模式之组合

    (文章翻译自Java Design Pattern: Composite) 组合模式相对来说是比较简单的,但是它在很多设计中被用到,比如SWT.Eclipse工作空间等等.它创建了一个可以通过唯一的方 ...

  4. 使用Oracle 9i工具管理数据库 - 初学者系列 - 学习者系列文章

    前面介绍了Oracle 9i的安装,本文大概介绍下Oracle 9i提供的管理工具的使用. 1 打开数据库配置工具 2 下一步 3 下一步 4 下一步.这里输入数据库名和SID 5 下一步 6 下一步 ...

  5. ubuntu下的apache+php+mysql的安装

    平时我都时在windows下搭配apache+php+mysql环境的,只不过后来听别人说在linux下搭配apache+php+mysql更受欢迎,而且一般公司也是用这样的搭配,所以今天在试着在ub ...

  6. asp.net访问WebService的各种方式

    WebService的访问形式主要有:SOAP调用.XMLHTTP POST.GET调用.MicroSoft.XMLDOMC调用.webbehavior.htc调用 我们知道的在C#后台本地调用Web ...

  7. Solr 教程

    1.Solr安装 下载jdk-8u111-windows-i586_8.0.1110.14 下载solr-6.3.0.zip 2.配置JAVA_HOME 在"系统变量"中,设置3项 ...

  8. 站点维护使用app_offline.htm页面提供友好的更新提示

    进行站点维护时为了以一个友好的方式提示给用户,比如什么“本网站正在更新”等等的信息可以建立一个叫app_offline.htm 的静态HTM页面文件,其中修改成你要临时显示的内容,将其放在你的应用的根 ...

  9. 开源框架之TAB控件

    我的开源框架之TAB控件   需求 (1)支持iframe.html.json格式的tab内容远程请求 (2)支持动态添加tab (3)支持远程加载完成监听,支持tab激活事件监听 (4)支持relo ...

  10. {{angular.js 使用技巧}} - 基于验证框架的扩展(w5cValidator)

    开场白: angular.js 是谷歌出的前端js MV*框架,我也是今年做 worktile 的时候才开始接触的,起初技术选型的时候还准备使用 backbone(毕竟很多大公司在使用他,而且也是比较 ...