洛谷题库 P2569 [SCOI2010]股票交易 简略题解
设 表示第 天持有 支股票,并且当天进行了 持有 / 买 / 卖 操作,所能获得的最大的收益。很容易想到以下转移:
观察发现第三维度是没有意义的——每一次转移的时候对于前序状态都使用了第三维的全部子状态。所以我们直接压缩掉。
另外,还有一个优化:易知 已经包含了 等所有最优决策,所以我们不用到 天前再选一遍。
于是有优化后的转移:
易知当 确定时,为定值。又因为题目中有的限制,所以可以在顺推转移时将决策看作一个“滑动窗口”。
我们希望在合法的情况下, 最大, 最大。很明显,决策具有单调性。单调队列优化已箭在弦上。对于上述两种转移,可以分别通过从小到大、从大到小扫描,即时加入当前决策、删除超限决策即可。
最终算法的时间复杂度为。
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
#include <bits/stdc++.h> using namespace std; const int N = 2048; int n, max_p, w; int ap[N], bp[N], as[N], bs[N]; int dp[N][N], q[N], h, t, dat[N], ans, pos; int main () { /* 洛谷题库 P2569 [SCOI2010]股票交易 */ scanf ("%d%d%d", &n, &max_p, &w); for (int i = 1; i <= n; ++i) scanf ("%d%d%d%d", ap + i, bp + i, as + i, bs + i); memset (dp, -0x3f, sizeof dp); dp[0][0] = 0; for (int i = 1; i <= n; ++i) { for (int j = 0; j <= max_p; ++j) dp[i][j] = dp[i - 1][j]; pos = max (0, i - w - 1); // 今天卖了股票 if (pos == 0) goto SELL; h = t = 0; memset (dat, 0, sizeof dat); memset (q, 0, sizeof q); for (int j = max_p; j >= 0; --j) { // 维护决策的单调队列 dat[j] = dp[pos][j] + bp[i] * j; // 新的决策! while (h < t && q[h + 1] > j + bs[i]) ++h; // 删除超界决策 while (h < t && dat[q[t]] <= dat[j]) --t; q[++t] = j; dp[i][j] = max (-bp[i] * j + dat[q[h + 1]], dp[i][j]), ans = max (ans, dp[i][j]); } // 今天买了股票 SELL: h = t = 0; memset (dat, 0, sizeof dat); memset (q, 0, sizeof q); for (int j = 0; j <= max_p; ++j) { dat[j] = dp[pos][j] + ap[i] * j; while (h < t && q[h + 1] < j - as[i]) ++h; while (h < t && dat[q[t]] <= dat[j]) --t; q[++t] = j; dp[i][j] = max (-ap[i] * j + dat[q[h + 1]], dp[i][j]), ans = max (ans, dp[i][j]); } } printf ("%d", ans); return 0; }