题目传送门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. java----Java的栈,堆,代码,静态存储区的存储顺序和位置

    转载:https://blog.csdn.net/zhangbaoanhadoop/article/details/82193497

  2. MyBatis mysal 日报表,月,年报表的统计

    mysql 按日.周.月.年统计sql语句整理,实现报表统计可视化 原文地址:http://blog.csdn.net/u010543785/article/details/52354957 最近在做 ...

  3. 数据结构C++实现代码-顺序表

    参考:https://blog.csdn.net/ebowtang/article/details/43094041 //seqList.h// //包含顺序表中的声明// #include<i ...

  4. 没有系列化导致错误:java.io.NotSerializableException: com.bjpowernode.bean.Team

    java.io.NotSerializableException: com.bjpowernode.bean.Team Cause: java.io.NotSerializableException: ...

  5. Sql Server并发和事务

    锁的作用范围通常在事务中,事务是建立在并发模式下. 从SQL Server 2005开始,加入了一种新的并发模式-----乐观并发.不管使用哪种并发模式,如果多个会话同时修改相同的数据,都会产生资源争 ...

  6. jmeter4.x centos7部署笔记

    1. jmeter依赖 java8或以上版本 安装 java : 参考  https://tecadmin.net/install-java-8-on-centos-rhel-and-fedora/ ...

  7. wb 黑名单批量操作

    0. 参考 yu961549745/WeiboBlackList  微博批量拉黑 1. 代码 block.py 更新内容:多线程,urllib.request 改为 requests + sessio ...

  8. 一起学Hadoop——Hadoop的前世今生

    Hadoop是什么? Hadoop是一个处理海量数据的开源框架.2002年Nutch项目面世,这是一个爬取网页工具和搜索引擎系统,和其他众多的工具一样,都遇到了在处理海量数据时效率低下,无法存储爬取网 ...

  9. 再理解tcp backlog

    在Linux 2.2以前,backlog大小包括了半连接状态和全连接状态两种队列大小.linux 2.2以后,分离为两个backlog来分别限制半连接SYN_RCVD状态的未完成连接队列大小跟全连接E ...

  10. ifconf家族命令

    1  ifconfig 命令: ifconfig 命令用来查看和配置网络设备.当网络环境发生改变时可通过此命令对网络进行相应的配置. 查看: ifconfig : 显示正在激活中的网卡 ifconfi ...