http://acm.hdu.edu.cn/showproblem.php?pid=3401

【题意】

  • 知道之后n天的股票买卖价格(api,bpi),以及每天股票买卖数量上限(asi,bsi),问他最多能赚多少钱。开始时有无限本金,要求任两次交易需要间隔W天以上,即第i天交易,第i+w+1天才能再交易。同时他任意时刻最多只能拥有maxp的股票

【思路】

  • dp[i][j]表示第i天拥有j支股票的最大收益,有三种转移方案:
  • dp[i][j]=max(dp[i][j],dp[i-1][j])表示第i天不买也不卖,由前一天转移而来
  • dp[i][j]=max(dp[i][j],dp[i-w-1][k]-(j-k)*ap[i])表示第i天买股票,有第i-w-1天转移而来
  • dp[i][j]=max(dp[i][j],dp[i-w-1][k]+(k-j)*bp[i])表示第i天卖股票,有第i-w-1天转移而来
  • 注意只需计算由i-w-1天转移而来,因为i-w-1天前的最优值已经通过不买不卖转移到了i-w-1天,即dp[i][j],j固定是随i单调递增的
  • 现在dp的复杂度是n^3,怎样降低复杂度?
  • 分析买股票的情况,dp[i][j]=max(dp[i-w-1][k]+k*ap[i])-j*ap[j],类似a[i]=max(b[k])+c[i],可以用单调队列优化
  • 我理解的是,状态数为2D,转移为1D,然后又有单调性,可以固定一维状态,把转移均摊到另一维,相当于转移是O(1)的,所以单调队列可以把dp降一维
  • a[i]=max(b[k]),若k<=j是从前往后递推,若k>=j是从后往前递推

【AC】

 #include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn=2e3+;
const int inf=0x3f3f3f3f;
int ap[maxn],bp[maxn],as[maxn],bs[maxn];
int n,maxp,w;
int dp[maxn][maxn];
struct node
{
int x;
int num;
}q[maxn];
int main()
{
int T;
scanf("%d",&T);
while(T--)
{
memset(dp,-inf,sizeof(dp));//求最大值,所以初始化为无穷小
scanf("%d%d%d",&n,&maxp,&w);
for(int i=;i<=n;i++) scanf("%d%d%d%d",&ap[i],&bp[i],&as[i],&bs[i]);
for(int i=;i<=n;i++) dp[i][]=;//拥有股票为0的最大收益当前是0
//前w+1天和[w+2,n]要分开算
//前w+1天只有两种情况:1.每天都不买不卖 2.其中一天买了股票 不能卖股票,而且最多只有一天能交易
for(int i=;i<=w+;i++)
{
for(int j=;j<=as[i];j++)
{
dp[i][j]=-j*ap[i];//可以选择在当天买
}
}
for(int j=;j<=maxp;j++)
{
for(int i=;i<=w+;i++)
{
dp[i][j]=max(dp[i][j],dp[i-][j]);//也可以选择由前一天转移而来
}
}
for(int i=w+;i<=n;i++)
{
int head=,tail=;
for(int j=;j<=maxp;j++)
{
dp[i][j]=max(dp[i][j],dp[i-][j]);//不买也不卖
//dp[i][j]=max(dp[i-w-1][k]+k*ap[i])-j*ap[i],其中k<=j
while(head<=tail&&q[tail].x<=dp[i-w-][j]+j*ap[i]) tail--;
q[++tail].x=dp[i-w-][j]+j*ap[i];q[tail].num=j;
while(q[head].num+as[i]<j) head++;
dp[i][j]=max(dp[i][j],q[head].x-j*ap[i]);
}
//dp[i][j]=max(dp[i-w-1][k]+k*bp[i])-j*bp[i],其中k>=j
head=,tail=;
for(int j=maxp;j>=;j--)
{
while(head<=tail&&q[tail].x<=dp[i-w-][j]+j*bp[i]) tail--;
q[++tail].x=dp[i-w-][j]+j*bp[i];q[tail].num=j;
while(q[head].num>bs[i]+j) head++;
dp[i][j]=max(dp[i][j],q[head].x-j*bp[i]);
}
}
int ans=;
for(int i=;i<=maxp;i++)
{
ans=max(ans,dp[n][i]);
}
printf("%d\n",ans);
} return ;
}

单调队列优化dp

【单调队列优化dp】HDU 3401 Trade的更多相关文章

  1. 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 ...

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

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

  3. 单调队列优化DP——习题收集

    前言 感觉可以用单调队列优化dp的模型还是挺活的,开个随笔记录一些遇到的比较有代表性的模型,断续更新.主要做一个收集整理总结工作. 记录 0x01 POJ - 1821 Fence,比较适合入门的题, ...

  4. Parade(单调队列优化dp)

    题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=2490 Parade Time Limit: 4000/2000 MS (Java/Others)    ...

  5. 1855: [Scoi2010]股票交易[单调队列优化DP]

    1855: [Scoi2010]股票交易 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 1083  Solved: 519[Submit][Status] ...

  6. bzoj1855: [Scoi2010]股票交易--单调队列优化DP

    单调队列优化DP的模板题 不难列出DP方程: 对于买入的情况 由于dp[i][j]=max{dp[i-w-1][k]+k*Ap[i]-j*Ap[i]} AP[i]*j是固定的,在队列中维护dp[i-w ...

  7. hdu3401:单调队列优化dp

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

  8. BZOJ_3831_[Poi2014]Little Bird_单调队列优化DP

    BZOJ_3831_[Poi2014]Little Bird_单调队列优化DP Description 有一排n棵树,第i棵树的高度是Di. MHY要从第一棵树到第n棵树去找他的妹子玩. 如果MHY在 ...

  9. 【单调队列优化dp】 分组

    [单调队列优化dp] 分组 >>>>题目 [题目] 给定一行n个非负整数,现在你可以选择其中若干个数,但不能有连续k个数被选择.你的任务是使得选出的数字的和最大 [输入格式] ...

随机推荐

  1. 远程linux服务器mysql数据库定期备份和删除

    网上已经有部分关于Linux下定期备份mysql的方法,但是很多步骤不够详细,不适合新手,自己琢磨了很久,终于搞定了. 1.Linux服务器一般是ssh协议,如果本地也是Linux环境,可以直接通过s ...

  2. Spring MVC能响应HTTP请求的原因?

    很多Java面试官喜欢问这个问题: 一个Spring MVC的项目文件里,开发人员没有开发自己的Servlet,只通过注解@RequestMapping定义了方法home能响应发向 /mvc/test ...

  3. Bellman-Ford与SPFA

    一.Bellman-Ford Bellman-Ford 算法是一种用于计算带权有向图中单源最短路径(当然也可以是无向图).与Dijkstra相比的优点是,也适合存在负权的图. 若存在最短路(不含负环时 ...

  4. EMVS: Event-based Multi-View Stereo 阅读笔记

    0. 摘要 EMVS目的:从已知轨迹的event相机,估计半稠密的3D结构 传统的MVS算法目的:从已知视点的图片集,去估计场景的稠密3D结构. EMVS2个固有属性: (1)   当传感器发生相对运 ...

  5. 第1节 flume:13、14、更多flume案例一,通过拦截器实现不同类型的数据区分

    1.6.flume案例一 1. 案例场景 A.B两台日志服务机器实时生产日志主要类型为access.log.nginx.log.web.log 现在要求: 把A.B 机器中的access.log.ng ...

  6. baidumap demo(一)

    覆盖物概述 地图上自定义的标注点和覆盖物我们统称为地图覆盖物.您可以通过定制BMKAnnotation和BMKOverlay来添加对应的标注点和覆盖物.地图覆盖物的设计遵循数据与View分离的原则,B ...

  7. ios之自定义导航栏上的返回按钮

    导航栏的按钮,右边的按钮是可以自己随意添加的.但左边的返回按钮怎么定制?你会说,添加一个自己的按钮呗!你可以试试看,这样行不行. 正确的答案是重载UINavigationController类的pus ...

  8. js基本语法之 值类型(数据类型)(变量类型)

    一.不可改变的原始值(栈数据)(五个) 数字(number),字符串(string),布尔值(boolean),undefined,null 其中;undefined是未定义的意思,而null是空的意 ...

  9. ABAQUS用户子程序一览表

    说明 ABAQUS用户子程序一览表 ABAQUSStandard subroutines Refence 说明 本系列文章本人基本没有原创贡献,都是在学习过程中找到的相关书籍和教程相关内容的汇总和梳理 ...

  10. Python使用三种方法实现PCA算法[转]

    主成分分析(PCA) vs 多元判别式分析(MDA) PCA和MDA都是线性变换的方法,二者关系密切.在PCA中,我们寻找数据集中最大化方差的成分,在MDA中,我们对类间最大散布的方向更感兴趣. 一句 ...