此题链接到dp常见优化方法

开始的时候被纪念品误导,以为是多支股票,后来发现事情不妙:

这道题知道的是某一只股票的走势;

\(Solution\):

\(70pts\):

设\(f[i][j]\)表示第i天持有j股股票可以获得的最大利润,有如下转移方程:

\(f[i][j]=Max\begin{cases}f[i-1][j] \ \ \ \ //第i天不买也不卖 \\-Ap[i]*j\ \ \ // 第i天之前不持有股票,第i天买入j股股票 \\f[i-w-1][k]-Ap[i]*(j-k) \ \ \ Max(0,j-As[i])\leq k< j \ \ //第i天买入j-k股股票 \\f[i-w-1][k]+Bp[i]*(k-j)\ \ \ j<k\leq Min(MaxP,j+Bs[i]);\ \ //第i天卖出k-j股股票 \end{cases}\)

为什么转移是第\(i-w-1\)天而不是第\(i-w-2\)亦或其它日子呢?

借用一段话(原文

看第一种情况(第i天不买也不卖),我们已经把某一天以前的最优答案转移到了该天,所以从那一天转移,相当于从那一天包括前面任何一天开始转移,省去了大把时间。

利用上面的四个式子就可以写出代码,注意初始化的时候要将\(f[ \ ][ \ ]\)初始化为一个很小的数,\(f[0][0]=0\)

\(Code:\)

#include<bits/stdc++.h>

using namespace std;

inline int read() {
int ans=0;
char last=' ',ch=getchar();
while(ch>'9'||ch<'0') last=ch,ch=getchar();
while(ch>='0'&&ch<='9') ans=(ans<<1)+(ans<<3)+ch-'0',ch=getchar();
if(last=='-') ans=-ans;
return ans;
}const int mxt=5010; int T,MaxP,w;
int Ap[mxt],Bp[mxt],As[mxt],Bs[mxt];
int f[mxt][mxt]; int main() {
T=read();
MaxP=read();
w=read();
for(int i=1;i<=T;i++) {
Ap[i]=read();
Bp[i]=read();
As[i]=read();
Bs[i]=read();
}
memset(f,0x9f,sizeof(f));
f[0][0]=0;
for(int i=1;i<=T;i++) {
for(int j=0;j<=MaxP;j++) {
f[i][j]=f[i-1][j];
if(i-w-1>=0) {
for(int k=max(0,j-As[i]);k<j;k++)
f[i][j]=max(f[i][j],f[i-w-1][k]-Ap[i]*(j-k));
for(int k=j+1;k<=min(MaxP,j+Bs[i]);k++) {
f[i][j]=max(f[i][j],f[i-w-1][k]+Bp[i]*(k-j));
}
}
if(j<=As[i])
f[i][j]=max(f[i][j],-Ap[i]*j);
printf("%d ",f[i][j]);
}
puts("");
}
printf("%d",f[T][0]);
return 0;
}

但不要以为这样就可以愉快的\(\mathfrak{AC}\)这道题,它毕竟是一道省选题\(\color{white}{(虽然年代已经很久远了}\)

实际复杂度\(O(T*MaxP^2)\),在\(T\)和\(MaxP\)都是\(2000\)的数量级的数据下,显然是过不去的(不要妄想跑过\(8e9\)

因此我们需要优化

看数据范围\(2000\)大致可以猜测是一个\(O(n^2)\)的代码

观察第\(3、4\)个式子,暴力拆解得到:

\(f[i][j]=Max\{f[i-w-1][k]-Ap[i]*j+Ap[i]*k\}\)

\(f[i][j]=Max\{f[i-w-1][k]-Bp[i]*j+Bp[i]*k\}\)

显然\(-Bp[i]*j\)与\(-Ap[i]*j\)的值不影响我们取最大值,因此我们可以提出来:

\(f[i][j]=Max\{f[i-w-1][k]+Ap[i]*k\}-Ap[i]*j\)

\(f[i][j]=Max\{f[i-w-1][k]+Bp[i]*k\}-Bp[i]*j\)

这两个式子像什么?像极了单调队列优化!

所以采用单调队列优化我们的第\(3、4\)个式子,就可以做到\(O(n^2)\)完成这道题(胜利的曙光\(\color{GOld}{!!!}\)

	h=1;t=0;
for(int j=0;j<=MaxP;j++) {//购买股票,从小往大更新
while(h<=t&&q[h]<j-As[i])
//清除队首中不符合"第i天的一次买入至多只能购买AS_i股"的
h++;
while(h<=t&&cost1(i,q[t])<=cost1(i,j))
//如果队尾的收入<=当前的收入,弹出队尾(从前往后更新,i要大于q[t],如果队尾的收入<=当前的收入,显然i相比于q[t]一定更优)
t--;
q[++t]=j;
if(h<=t) //队列中有元素
f[i][j]=max(f[i][j],cost1(i,q[h])-Ap[i]*j);
}
h=1;t=0;
for(int j=MaxP;j>=0;j--) {//卖出股票,从大往小更新:由4式知,j位置的状态是从一个>j的位置转移而来的,因此倒序更新
while(h<=t&&q[h]>j+Bs[i])
h++;
while(h<=t&&cost2(i,q[t])<=cost2(i,j))
t--;
q[++t]=j;
if(h<=t)
f[i][j]=max(f[i][j],cost2(i,q[h])-Bp[i]*j);
}

\(Code:\)

#include<bits/stdc++.h>

using namespace std;

inline int read() {
int ans=0;
char last=' ',ch=getchar();
while(ch>'9'||ch<'0') last=ch,ch=getchar();
while(ch>='0'&&ch<='9') ans=(ans<<1)+(ans<<3)+ch-'0',ch=getchar();
if(last=='-') ans=-ans;
return ans;
}const int mxt=5010; int T,MaxP,w;
int Ap[mxt],Bp[mxt],As[mxt],Bs[mxt];
int f[mxt][mxt];
int q[mxt];
int h,t; int cost1(int i,int j) {
return f[i-w-1][j]+Ap[i]*j;
} int cost2(int i,int j) {
return f[i-w-1][j]+Bp[i]*j;
} int main() {
T=read();
MaxP=read();
w=read();
for(int i=1;i<=T;i++) {
Ap[i]=read();
Bp[i]=read();
As[i]=read();
Bs[i]=read();
}
memset(f,0x9f,sizeof(f));
f[0][0]=0; for(int i=1;i<=T;i++) {
for(int j=0;j<=MaxP;j++) {
f[i][j]=f[i-1][j];
if(j<=As[i])
f[i][j]=max(f[i][j],-Ap[i]*j);
}
if(i-w<=0) continue;
h=1;t=0;
for(int j=0;j<=MaxP;j++) {
while(h<=t&&q[h]<j-As[i]) h++;
while(h<=t&&cost1(i,q[t])<=cost1(i,j)) t--;
q[++t]=j;
if(h<=t)
f[i][j]=max(f[i][j],cost1(i,q[h])-Ap[i]*j);
}
h=1;t=0;
for(int j=MaxP;j>=0;j--) {
while(h<=t&&q[h]>j+Bs[i]) h++;
while(h<=t&&cost2(i,q[t])<=cost2(i,j)) t--;
q[++t]=j;
if(h<=t)
f[i][j]=max(f[i][j],cost2(i,q[h])-Bp[i]*j);
}
}
printf("%d",f[T][0]);
return 0;
}

Luogu P2569 [SCOI2010] 股票交易的更多相关文章

  1. [luogu] P2569 [SCOI2010]股票交易 (单调队列优化)

    P2569 [SCOI2010]股票交易 题目描述 最近 \(\text{lxhgww}\) 又迷上了投资股票,通过一段时间的观察和学习,他总结出了股票行情的一些规律. 通过一段时间的观察,\(\te ...

  2. LUOGU P2569 [SCOI2010]股票交易(单调队列优化dp)

    传送门 解题思路 不难想一个\(O(n^3)\)的\(dp\),设\(f_{i,j}\)表示第\(i\)天,手上有\(j\)股的最大收益,因为这个\(dp\)具有单调性,所以\(f_i\)可以贪心的直 ...

  3. 洛谷P2569 [SCOI2010]股票交易

    P2569 [SCOI2010]股票交易 题目描述 最近lxhgww又迷上了投资股票,通过一段时间的观察和学习,他总结出了股票行情的一些规律. 通过一段时间的观察,lxhgww预测到了未来T天内某只股 ...

  4. P2569 [SCOI2010]股票交易 dp 单调队列优化

    LINK:股票交易 题目确实不算难 但是坑点挺多 关于初值的处理问题我就wa了两次. 所以来谢罪. 由于在手中的邮票的数量存在限制 且每次买入卖出也有限制. 必然要多开一维来存每天的邮票数量. 那么容 ...

  5. P2569 [SCOI2010]股票交易

    最近 \(lxhgww\) 又迷上了投资股票,通过一段时间的观察和学习,他总结出了股票行情的一些规律. 通过一段时间的观察,\(lxhgww\) 预测到了未来 \(T\) 天内某只股票的走势,第 \( ...

  6. 洛谷 P2569[SCOI2010]股票交易(动规+单调队列)

    //只能写出裸的动规,为什么会有人能想到用单调队列优化Orz 题目描述 最近lxhgww又迷上了投资股票,通过一段时间的观察和学习,他总结出了股票行情的一些规律. 通过一段时间的观察,lxhgww预测 ...

  7. Luogu 2569 [SCOI2010]股票交易 (朴素动规转移 + 单调队列优化)

    题意: 已知未来 N 天的股票走势,第 i 天最多买进 as [ i ] 股每股 ap [ i ] 元,最多卖出 bs [ i ] 股每股 bp [ i ] 元,且每天最多拥有 Mp 股,且每两次交易 ...

  8. 洛谷P2569 [SCOI2010]股票交易(单调队列)

    传送门 惭愧……这种题目都没看出来…… 首先,我们用$dp[i][j]$表示在第$i$天,手上有$j$股时的最大收益 第一,我们可以直接买股票,即$dp[i][j]=-j*AP_i$,这个直接计算即可 ...

  9. 单调队列优化DP || [SCOI2010]股票交易 || BZOJ 1855 || Luogu P2569

    题面:P2569 [SCOI2010]股票交易 题解: F[i][j]表示前i天,目前手中有j股的最大收入Case 1:第i天是第一次购买股票F[i][j]=-j*AP[i]; (1<=j< ...

随机推荐

  1. noi.ac NOI挑战营模拟赛1-5

    注:因为博主是个每次考试都爆零垫底的菜鸡,所以此篇博客很有可能咕咕咕 (指只贴AC代码不写题解的......如果我真的不会做的话,就不能怪我了qwqwq) Day1 T1 swap 23pts 从一个 ...

  2. 原生Js_简易图片轮播模板

    功能:图片自动循环轮播,通过点击“上一张”,“下一张”按钮来控制图片的切换 <!DOCTYPE html> <html> <head> <meta chars ...

  3. Java - 自动装箱与拆箱详解

    1.装箱与拆箱 装箱,将基本数据类型转为包装类型.拆箱,将包装类型转为基本数据类型. // Byte, Short, Integer, Long, Double, Float, Boolean, Ch ...

  4. docker启动常见报错

    Docker启动时的报错汇总 22017.11.10 16:30:29字数 575阅读 27184 八个Docker常见故障 https://mp.weixin.qq.com/s/2GNKmRJtBG ...

  5. Windows 下的SSH客户端

    在日常Linux系统管理中,会使用SSH工具连接服务器,之所以SSH连接主要是为了安全,传统的telnet连接方式是以明文传输,很不安全,网络中如果又热窃听抓包,密码将要泄露.在众多SSH连接中,Pu ...

  6. struts2数据处理的几种方式

    package com.loaderman.c_data; import java.util.Map; import javax.servlet.ServletContext; import com. ...

  7. python处理文件---每行末尾加上字符

    题记:该代码是项目中需要处理一个文件,所以使用python去处理,还有另外一个方法就是使用notepad++的正则替换,这里就只是使用python脚本去处理了. 我想只要你懂一门语言,就很容易处理这个 ...

  8. Selenium 2自动化测试实战30(unittest补充)

    unittest补充 1.用例执行的顺序 #test.py #coding:utf-8 from Demo1 import Count import unittest class TestBdd(un ...

  9. 小D课堂 - 零基础入门SpringBoot2.X到实战_第14节 高级篇幅之SpringBoot多环境配置_59、SpringBoot多环境配置介绍和项目实战

    笔记 1.SpringBoot多环境配置介绍和项目实战(核心知识)     简介:SpringBoot介绍多环境配置和使用场景 1.不同环境使用不同配置         例如数据库配置,在开发的时候, ...

  10. 阶段3 3.SpringMVC·_07.SSM整合案例_05.ssm整合之Spring整合SpringMVC的框架

    点击超连接,执行controller里面的方法 那么就需要在Controller里面定义Service对象,就需要依赖注入进来. 启动tomcat服务器,web.xml里面的前端控制器会帮我加载spr ...