题目传送门https://www.luogu.org/problemnew/show/P2569

第一眼看题就觉得是个dp ,然后看到2000的范围,hmm大概是个n^2的2维dp

开始设状态,第一维肯定是天数的枚举,第二维...看着MaxP的范围,觉得应该是持有股票数量的枚举

那么dp[i][j]就是对于1~i天,第i天持有j张股票的情况下最多能赚的钱

然后想转移方程,分成几种情况

case 1:前面都不买,从第i天开始买

  也就是从一无所有的情况下买j张股票dp[i][j]=-APi*j

case 2:不买不卖

  这种情况下不会就不会受到W的限制,直接dp[i][j]=dp[i-1][j]

case 3:发生买卖

  首先要考虑应该从前面哪一天的状态转移过来,也就是上一次买卖在哪一天发生。

  乍一看好像没办法知道应该从哪一天转移,莫非又多一层1~i-W-1的循环?

  其实不用。上一次买卖所获得的最大值可以根据case 2的dp方程向后转移

  所以直接从i-W-1天转移就行

  case 3.1:又卖又买

    仔细想想,这种情况不可能成为最大值。因为有APi>=BPi,卖了又买不如买少点,多折腾是不行的

  case 3.2:只买不卖

    既然是买,那么之前持有的股票必然比j少,枚举之前持有的股票k,花钱(j-k)*APi

    dp[i][j]=dp[i-W-1][k]-(j-k)*APi;

  case3.3:只卖不买

    与case 3.2类似

    dp[i][j]=max dp[i-W-1][k]+(k-j)*BPi

大概就是这样。dp[T][0]就是最终答案(保险起见我写的是max dp[T][0~MaxP])

时间大概是O(n*MaxP^2)

预期50pt做法(实际70pt)

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
using namespace std; int T,MaxP,W; struct pig {
int APi,BPi,ASi,BSi;
} day[]; int dp[][]; int main() {
cin>>T>>MaxP>>W;
for(int i=; i<=T; i++) {
scanf("%d%d%d%d",&day[i].APi,&day[i].BPi,&day[i].ASi,&day[i].BSi);
}
for(int i=; i<=T; i++) {
for(int j=; j<=MaxP; j++) {
dp[i][j]=-;
}
}
dp[][]=; for(int i=;i<=T;i++){
for(int j=;j<=MaxP;j++){
dp[i][j]=dp[i-][j];
if(j<=day[i].ASi){ dp[i][j]=max(dp[i][j],-j*day[i].APi);
}
if(i-W-<){
continue;
}
for(int k=j-day[i].ASi;k<j;k++){
if(k<){
continue;
}
dp[i][j]=max(dp[i][j],dp[i-W-][k]-(j-k)*day[i].APi);
}
for(int k=j+;k<=j+day[i].BSi;k++){
if(k>MaxP){
break;
}
dp[i][j]=max(dp[i][j],dp[i-W-][k]+(k-j)*day[i].BPi);
}
}
}
int ans=;
for(int j=; j<=MaxP; j++) {
ans=max(ans,dp[T][j]);
}
cout<<ans;
return ;
}
/*
输入
初始化 赋-inf,dp[0][0]=0
dp循环
i 天数1-T
j 当前持股数0-Maxp
从当前天开买 j<=Asi dp[i][j]=-j*APi;
不买不卖 dp[i][j]=dp[i-1][j]
如果i-W-1>=0
k 前次购买时持股数 买股 (j-Asi)~j&&>=0
dp[i][j]=max dp[i-W-1][k]-(j-k)*APi;
k 前次购买时持股数 卖股 j~(j+Bsi)&&<=MaxP
dp[i][j]=max dp[i-W-1][k]+(k-j)*BPi
输出,max(dp[0][0~MaxP]) 自测数据
2 1
1 1 1
1 1 1
2 1 1
3 1 1
4 1 1 2 0
1 1 1
1 1 1
2 1 1
3 1 1
4 1 1
*/

(附伪代码提纲)


怎么继续优化呢?

如果仍然保持dp[i][j]的形式,i和j的循环肯定是不能去掉的,所以考虑优化k

用case 3.2:只买不卖 举例

它的dp方程是dp[i][j]=max dp[i-W-1][k]-(j-k)*APi;

乘法分配律得dp[i][j]=max dp[i-W-1][k]+k*APi-j*APi;

j是枚举出来的,所以可以拆分一下max,得dp[i][j]=max(dp[i-W-1][k]+k*APi)-j*APi;

因为对于每个确定的i和j,k属于区间 [j - ASi ,j-1] ,假设i是定值,ASi就是定值。那么对于所有j,k的区间大小是一样的

一个长度固定的区间一格一格的移动...这不就是滑动窗口

单调队列优化,O(n*MaxP),100pt到手

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
using namespace std; int T,MaxP,W; struct MonoQue {
int lim;//维持的长度
int size;//最大存储量
int que[];//单调队列
int data[];//维护的数据
int head,tail; void init(int mys,int mylim) {
size=mys;
lim=mylim;
head=;
tail=;
}
void push(int id) {//向单调队列中加入并维护,id可以在特殊情况下跳跃前进
if(id<=size) {//防越界处理
for(; head<tail;) {
if(data[que[tail-]]>data[id]) {
break;
}
tail--;
}
que[tail]=id;
tail++;
}
if(que[head]+lim<=id) {
pop();
}
}
void pop() {
head++;
}
int quetop() {
return que[head];
}
int top() {
return data[quetop()];
}
}; struct pig {
int APi,BPi,ASi,BSi;
} day[]; int dp[][]; int main() {
cin>>T>>MaxP>>W;
for(int i=; i<=T; i++) {
scanf("%d%d%d%d",&day[i].APi,&day[i].BPi,&day[i].ASi,&day[i].BSi);
}
for(int i=; i<=T; i++) {
for(int j=; j<=MaxP; j++) {
dp[i][j]=-;
}
}
dp[][]=; MonoQue buy,sell;
for(int i=; i<=T; i++) {
buy.init(MaxP,day[i].ASi+);
sell.init(MaxP,day[i].BSi+);
if(i-W->=) {//防数组超下界
for(int j=; j<=MaxP; j++) {
buy.data[j]=dp[i-W-][j]+j*day[i].APi;
sell.data[j]=dp[i-W-][j]+j*day[i].BPi;
}
}
for(int j=; j<day[i].BSi; j++) {
sell.push(j);//sell需要预先加入
}
for(int j=; j<=MaxP; j++) {
dp[i][j]=dp[i-][j]; if(j<=day[i].ASi) {
dp[i][j]=max(dp[i][j],-j*day[i].APi);
}
if(i-W-<) {
continue;
}
buy.push(j);
dp[i][j]=max(dp[i][j],buy.top()-j*day[i].APi); sell.push(j+day[i].BSi);//单调队列里会处理超了MaxP的情况 dp[i][j]=max(dp[i][j],sell.top()-j*day[i].BPi);
}
}
int ans=;
for(int j=; j<=MaxP; j++) {
ans=max(ans,dp[T][j]);
}
cout<<ans;
return ;
}
/*
自测数据
3 2 0
2 1 1 1
3 2 1 1
4 3 1 1
*/

参考:

https://www.luogu.org/blog/Sooke/solution-p2569

洛谷P2569 股票交易的更多相关文章

  1. 洛谷P2569 股票交易 [SCOI2010] dp

    正解:dp+单调队列优化 解题报告: 先放个传送门鸭qwq umm首先dp转移挺好想的?就买和不买 f[i][j]表示第i天手上有j的股份的最多钱,转移也很好想?就枚举第1天到第i-w-1天枚举买k股 ...

  2. 洛谷P2569 股票交易【dp】【单调队列】

    题目描述 最近 \text{lxhgww}lxhgww 又迷上了投资股票,通过一段时间的观察和学习,他总结出了股票行情的一些规律. 通过一段时间的观察,\text{lxhgww}lxhgww 预测到了 ...

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

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

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

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

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

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

  6. 洛谷P2569 (BZOJ1855)[SCOI2010]股票交易 【单调队列优化DP】

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

  7. 洛谷 [SCOI2010]股票交易 | 单调性DP

    题目链接 #include<cstdio> #include<algorithm> #include<cstring> #define N 2005 using n ...

  8. BZOJ1855或洛谷2569 [SCOI2010]股票交易

    一道单调队列优化\(DP\) BZOJ原题链接 洛谷原题链接 朴素的\(DP\)方程并不难想. 定义\(f[i][j]\)表示到第\(i\)天,手上持有\(j\)股时的最大收益. 转移方程可以分成四个 ...

  9. 洛谷1640 bzoj1854游戏 匈牙利就是又短又快

    bzoj炸了,靠离线版题目做了两道(过过样例什么的还是轻松的)但是交不了,正巧洛谷有个"大牛分站",就转回洛谷做题了 水题先行,一道傻逼匈牙利 其实本来的思路是搜索然后发现写出来类 ...

随机推荐

  1. name

    问题 A: name 时间限制: 1 Sec  内存限制: 256 MB 题目描述 lpq同学最近突然对外国人的名字产生了兴趣,特别是外国女生的名字,于是他开始试图去认识一些国外的女生. 随着认识的女 ...

  2. Java SimpleDateFormat 中英文时间格式化转换

    2015年08月29日 17:37:43 阅读数:32459 SimpleDateFormat是一个以与语言环境有关的方式来格式化和解析日期的具体类.它允许进行格式化(日期 -> 文本).解析( ...

  3. 数据库和linux对大小写的区分

  4. 十六进制的ASCII码 "\u6cf0\u56fd" 解码成unicode

    转码方法: C#: string a = "\u6cf0\u56fd"; string b = Encoding.UTF8.GetString(Encoding.UTF8.GetB ...

  5. 安全测试robots

    http://stock.pingan.com/robots.txt

  6. flink的集群的HA高可用

    对于一个企业级的应用,稳定性是首要要考虑的问题,然后才是性能,因此 HA 机制是必不可少的: 和 Hadoop 一代一样,从架构中我们可以很明显的发现 JobManager 有明显的单点问题(SPOF ...

  7. Python学习(二十七)—— Django和pymysql搭建学员管理系统

    转载自http://www.cnblogs.com/liwenzhou/p/8270250.html 一.学员管理系统 1.项目规划阶段 项目背景 近年来老男孩教育的入学学员数量稳步快速增长,传统的e ...

  8. Codeforces 1139D Steps to One dp

    Steps to One 啊, 我要死了, 这种垃圾题居然没写出来, 最后十分钟才发现错在哪. 不知道为什么我以为 对于一个数x , 除了它的因子和它的倍数都是和它互质的, 我脑子是抽了吗? 随便瞎d ...

  9. P1057 传球游戏 dp

    题目描述 上体育课的时候,小蛮的老师经常带着同学们一起做游戏.这次,老师带着同学们一起做传球游戏. 游戏规则是这样的:nn个同学站成一个圆圈,其中的一个同学手里拿着一个球,当老师吹哨子时开始传球,每个 ...

  10. Hive| DDL| DML

    类型转换 可以使用CAST操作显示进行数据类型转换 例如CAST(' 转换成整数1:如果强制类型转换失败,如执行CAST('X' AS INT),表达式返回空值 NULL. : jdbc:hive2: ...