题目链接

BZOJ1855

题解

设\(f[i][j]\)表示第\(i\)天结束时拥有\(j\)张股票时的最大收益

若\(i \le W\),显然在这之前不可能有交易

\[f[i][j] = max\{f[i - 1][j],-ap[i] * j\} \quad [j \le as[i]]
\]

否则,就有三种选择:

①购买

\[f[i][j] = max\{f[i - W - 1][k] - ap[i] * (j - k)\} \quad[k \le j][j - k \le as[i]]
\]

②卖出

\[f[i][j] = max\{f[i - W - 1][k] + bp[i] * (k - j)\} \quad[k \ge j][k - j \le bs[i]]
\]

③什么也不做

\[f[i][j] = max\{f[i][j],f[i - 1][j]\}
\]

其中③总共是\(O(n^2)\)的

①和②如果逐个枚举是\(O(n^3)\)的,无法承受

拆开式子可发现可以用单调队列优化成\(O(n^2)\)

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#define LL long long int
#define Redge(u) for (int k = h[u],to; k; k = ed[k].nxt)
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define cls(s) memset(s,-0x3f3f3f3f,sizeof(s))
using namespace std;
const int maxn = 2005,maxm = 100005,INF = 1000000000;
inline int read(){
int out = 0,flag = 1; char c = getchar();
while (c < 48 || c > 57){if (c == '-') flag = -1; c = getchar();}
while (c >= 48 && c <= 57){out = (out << 3) + (out << 1) + c - 48; c = getchar();}
return out * flag;
}
int f[maxn][maxn],T,P,W,ap[maxn],bp[maxn],as[maxn],bs[maxn];
struct node{
int k,v;
}q[maxn];
int head,tail;
int main(){
T = read(); P = read(); W = read();
REP(i,T) ap[i] = read(),bp[i] = read(),as[i] = read(),bs[i] = read();
cls(f); f[0][0] = 0; int ans = 0;
for (int i = 1; i <= T; i++){
for (int j = 0; j <= P; j++) f[i][j] = f[i - 1][j];
if (i <= W){
for (int j = 0; j <= as[i]; j++)
f[i][j] = max(f[i][j],-ap[i] * j);
}
else {
head = 0; tail = -1;
for (int j = 0; j <= P; j++){
while (head <= tail && (j - q[head].k) > as[i]) head++;
while (head <= tail && q[tail].v < f[i - W - 1][j] + ap[i] * j) tail--;
q[++tail] = (node){j,f[i - W - 1][j] + ap[i] * j};
f[i][j] = max(f[i][j],q[head].v - ap[i] * j);
}
head = 0; tail = -1;
for (int j = P; j >= 0; j--){
while (head <= tail && (q[head].k - j) > bs[i]) head++;
while (head <= tail && q[tail].v < f[i - W - 1][j] + bp[i] * j) tail--;
q[++tail] = (node){j,f[i - W - 1][j] + bp[i] * j};
f[i][j] = max(f[i][j],q[head].v - bp[i] * j);
}
}
ans = max(ans,f[i][0]);
}
printf("%d\n",ans);
return 0;
}

最新文章

  1. [转] vim自定义配置 和 在ubnetu中安装vim
  2. Entity Framework 摘记
  3. VR定制 AR定制 就找北京动软VR开发团队(VR案例 AR案例)
  4. SpringMVC 返回 html 视图页面,SpringMVC与Servlet,Servlet重定向与转发
  5. redis tcp-backlog配置
  6. jquery height
  7. FHS目录配置下,常见的几个问题及解答
  8. POJ C程序设计进阶 编程题#2: 配对碱基链
  9. a标签中使用img后的高度多了4px
  10. 解决Adobe Acrobat “正在纠偏图像,正在旋转图像,正在分解页面”问题
  11. 阿尔贝我给我加i觉
  12. 第2章 熟悉Eclipse开发工具---- System.out.println(&quot;sum=&quot;+(a+b));
  13. Unity UGUI基础之Image
  14. learning coap protocol
  15. Idea安装lombok插件【转载】
  16. jvm(转)
  17. libc.so.6: version &#39;GLIBC_2.14&#39; not found报错提示的解决方案
  18. Jmeter测试报告
  19. 【BZOJ4300】 绝世好题
  20. 论文翻译技巧--Notepad替换回车

热门文章

  1. 泉五培训Day1
  2. 学习python第十四天,模块
  3. C语言字符篇(四)字符串查找函数
  4. 裸机——LCD
  5. 使用MD5比较两个文件是否相同
  6. 您的手机上未安装应用程序 android 点击快捷方式提示未安装程序的解决
  7. windows系统如何查看某个端口被谁占用
  8. Windows下使用Nginx+tomcat配置负载均衡
  9. 机器学习tensorflow框架初试
  10. Android学习记录(10)—Android之图片颜色处理