题解 CF559E 【Gerald and Path】

xht

2019-12-07 15:54:02

Solution

> [CF559E Gerald and Path](https://codeforc.es/contest/559/problem/E) ## 题意 - 有 $n$ 条线段。 - 每条线段给定其中一端的位置及长度。 - 求所有线段覆盖的最大长度。 - $n \le 100$。 ## 题解 首先对一端位置排序。 考虑 dp,设 $f_{i,j,p}$ 表示考虑前 $i$ 条线段,最靠右的线段是第 $j$ 条,其方向为 $p$($0$ 左 $1$ 右)。 考虑如何转移,从小到大枚举 $i$ 后面的线段以及它的方向,同时维护枚举过的线段中最靠右的线段以及它的方向。 设线段 $j$ 在 $p$ 方向的右端点是 $o$;此时枚举到线段 $k$,它的方向为 $q$,长度为 $l$,右端点是 $t$;枚举过的线段中最靠右的线段是 $x$,它的方向为 $y$,右端点是 $z$。 那么有转移 $f_{k,x,y} \leftarrow f_{i,j,p} + \min(l, t - o) + z - t$。 时间复杂度 $\mathcal O(n^3)$。 ## 代码 ```cpp const int N = 107; int n, ans, f[N][N][2]; pi a[N]; int main() { rd(n); for (int i = 1; i <= n; i++) rd(a[i].fi), rd(a[i].se); sort(a + 1, a + n + 1); a[0].fi = -1e9; for (int i = 0; i <= n; i++) for (int j = 0; j <= i; j++) for (int p = 0; p < 2; p++) { ans = max(ans, f[i][j][p]); int o = a[j].fi + p * a[j].se, mx = -1e9, x, y; for (int k = i + 1; k <= n; k++) for (int q = 0; q < 2; q++) { int t = a[k].fi + q * a[k].se; if (t > mx) mx = t, x = k, y = q; f[k][x][y] = max(f[k][x][y], f[i][j][p] + min(a[k].se, t - o) + mx - t); } } print(ans); return 0; } ```