套路化了。
比较显然的关路灯型区间 dp。考虑你 \(t\) 时刻熄灭一个初始长度为 \(a\) 的蜡烛,得分是 \(\max(a - t, 0)\),所以尝试把时间塞进状态。设 \(f_{i, j, k, 0/1}\) 表示,熄灭完区间 \([i, j]\) 的蜡烛,当前时间是 \(t\),在左端点还是右端点的最大得分。转移考虑走的路径即可。
发现时间可能很大,并且已经没有优化的余地了。考虑换种思路,假设我一开始得了 \(\sum\limits_{i = 1}^n a_i\) 分,然后我每走一个单位距离,得分减少 \(k\),\(k\) 是还剩下的蜡烛数量。但是你发现一个很麻烦的事情是,熄灭蜡烛时得分要与 \(0\) 取 \(\max\),但是你到这个蜡烛的时候它的得分已经是负数了。
考虑直接钦定每个蜡烛的长度是否计入最后得分,称这样的蜡烛是有效的,这样如果遇到负数得分的蜡烛,最优解一定是不把它当成有效。状态添加一维 \(k\) 表示熄灭区间 \([i, j]\) 的蜡烛后还剩下 \(k\) 个有效蜡烛。转移就根据关路灯的基础上再讨论当前熄灭的这个蜡烛是否有效即可。
没了?
代码很好写。时间复杂度 \(O(n^3)\)。
code
// Problem: H - Candles
// Contest: AtCoder - Sciseed Programming Contest 2021(AtCoder Beginner Contest 219)
// URL: https://atcoder.jp/contests/abc219/tasks/abc219_h
// Memory Limit: 1024 MB
// Time Limit: 2000 ms
//
// Powered by CP Editor (https://cpeditor.org)
#include <bits/stdc++.h>
#define pb emplace_back
#define fst first
#define scd second
#define mems(a, x) memset((a), (x), sizeof(a))
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
typedef double db;
typedef long double ldb;
typedef pair<ll, ll> pii;
const int maxn = 310;
ll n, m, f[maxn][maxn][maxn][2];
struct node {
ll x, y;
node(ll a = 0, ll b = 0) : x(a), y(b) {}
} a[maxn];
void solve() {
scanf("%lld", &n);
for (int i = 1; i <= n; ++i) {
scanf("%lld%lld", &a[i].x, &a[i].y);
}
a[++n] = node(0, 0);
sort(a + 1, a + n + 1, [&](node a, node b) {
return a.x < b.x;
});
int p = -1;
for (int i = 1; i <= n; ++i) {
if (!a[i].y) {
p = i;
break;
}
}
mems(f, -0x3f);
for (int i = 0; i <= n; ++i) {
f[p][p][i][0] = f[p][p][i][1] = 0;
}
for (int p = 2; p <= n; ++p) {
for (int i = 1, j = p; j <= n; ++i, ++j) {
for (int k = 0; k <= n; ++k) {
f[i][j][k][0] = max({f[i + 1][j][k][0] - (a[i + 1].x - a[i].x) * k, f[i + 1][j][k + 1][0] - (a[i + 1].x - a[i].x) * (k + 1) + a[i].y, f[i + 1][j][k][1] - (a[j].x - a[i].x) * k, f[i + 1][j][k + 1][1] - (a[j].x - a[i].x) * (k + 1) + a[i].y});
f[i][j][k][1] = max({f[i][j - 1][k][0] - (a[j].x - a[i].x) * k, f[i][j - 1][k + 1][0] - (a[j].x - a[i].x) * (k + 1) + a[j].y, f[i][j - 1][k][1] - (a[j].x - a[j - 1].x) * k, f[i][j - 1][k + 1][1] - (a[j].x - a[j - 1].x) * (k + 1) + a[j].y});
}
}
}
ll ans = 0;
for (int i = 1; i <= n; ++i) {
for (int j = i; j <= n; ++j) {
for (int k = 0; k <= n; ++k) {
ans = max({ans, f[i][j][k][0], f[i][j][k][1]});
}
}
}
printf("%lld\n", ans);
}
int main() {
int T = 1;
// scanf("%d", &T);
while (T--) {
solve();
}
return 0;
}