bzoj 1855 dp + 单调队列优化
思路:很容易写出dp方程,很容易看出能用单调队列优化。。
#include<bits/stdc++.h> #define LL long long #define fi first #define se second #define mk make_pair #define PII pair<int, int> #define y1 skldjfskldjg #define y2 skldfjsklejg using namespace std; const int N = 2000 + 7; const int inf = 0x3f3f3f3f; const LL INF = 0x3f3f3f3f3f3f3f3f; const int mod = 10000; int n, m, w, ap[N], bp[N], as[N], bs[N], dp[N][N]; int stk[N], head, rear; int cal1(int p, int i, int x) { return dp[p][x] + x * ap[i]; } int cal2(int p, int i, int x) { return dp[p][x] + x * bp[i]; } int main() { scanf("%d%d%d", &n, &m, &w); for(int i = 1; i <= n; i++) scanf("%d%d%d%d", &ap[i], &bp[i], &as[i], &bs[i]); for(int i = 0; i <= n; i++) for(int j = 0; j <= m; j++) dp[i][j] = -inf; dp[0][0] = 0; int ans = 0; for(int i = 1; i <= n; i++) { head = 1, rear = 0; int p = max(0, i - w - 1); for(int j = 0; j <= m; j++) dp[i][j] = dp[i - 1][j]; for(int j = 0; j <= m; j++) { while(rear >= head && cal1(p, i, j) > cal1(p, i, stk[rear])) rear--; stk[++rear] = j; while(j - as[i] > stk[head]) head++; dp[i][j] = max(dp[i][j], cal1(p, i, stk[head]) - j * ap[i]); } head = 1, rear = 0; for(int j = m; j >= 0; j--) { while(rear >= head && cal2(p, i, j) > cal2(p, i, stk[rear])) rear--; stk[++rear] = j; while(j + bs[i] < stk[head]) head++; dp[i][j] = max(dp[i][j], cal2(p, i, stk[head]) - j * bp[i]); ans = max(ans, dp[i][j]); } } printf("%d\n", ans); return 0; } /* */