|
发表于 2025-8-10 22:02:20
|
显示全部楼层
根据你的描述,你正在学习斜率优化,并尝试解决任务安排问题(P5785)。你的推导思路正确,但代码存在一些问题。以下是修正后的简洁解答:
### 斜率优化核心思路
1. **状态转移方程**:
`dp[i] = min{ dp[j] + sumT[i]*(sumC[i]-sumC[j]) + s*(sumC[n]-sumC[j]) }`
其中 `sumT[i] = Σt[1..i]`, `sumC[i] = Σc[1..i]`。
2. **斜率优化转换**:
- 将方程变形为:`dp[j] = (sumT[i] + s) * sumC[j] + (dp[i] - sumT[i]*sumC[i] - s*sumC[n])`
- 令 `y = dp[j]`, `x = sumC[j]`, `k_i = sumT[i] + s`
- 目标:找到点 `(x, y)` 使截距最小(即 `dp[i]` 最小)
3. **维护下凸包**:
- 用单调队列维护候选点集,保证相邻点斜率递增(下凸包性质)。
- 由于 `k_i` 不单调(`t_i` 可能为负),需在凸包上二分查找第一个斜率 `≥k_i` 的线段左端点。
### 修正代码(附注释)
- #include <iostream>
- #define int long long
- #define rep(i, a, b) for (int i = a; i <= b; i++)
- using namespace std;
- const int N = 3e5 + 10;
- int n, s;
- int sumT[N], sumC[N], dp[N], q[N];
- // 斜率比较:判断 (q[r-1],q[r]) 与 (q[r],i) 的斜率大小(避免浮点)
- bool check(int a, int b, int c) {
- int x1 = sumC[b] - sumC[a], y1 = dp[b] - dp[a];
- int x2 = sumC[c] - sumC[b], y2 = dp[c] - dp[b];
- return y1 * x2 >= y2 * x1; // 若成立,则删除队尾b点
- }
- signed main() {
- cin >> n >> s;
- rep(i, 1, n) {
- int t, c;
- cin >> t >> c;
- sumT[i] = sumT[i - 1] + t;
- sumC[i] = sumC[i - 1] + c;
- }
- int l = 0, r = 0; // 队列初始化(q[0]=0)
- q[0] = 0;
- rep(i, 1, n) {
- int k = sumT[i] + s; // 当前斜率k_i
-
- // 二分查找最优决策点
- int L = l, R = r;
- while (L < R) {
- int mid = (L + R) >> 1;
- // 比较 mid 与 mid+1 的斜率
- if (dp[q[mid + 1]] - dp[q[mid]] >= k * (sumC[q[mid + 1]] - sumC[q[mid]]))
- R = mid;
- else
- L = mid + 1;
- }
- int j = q[L]; // 最优决策点j
-
- // 更新dp[i]
- dp[i] = dp[j] + sumT[i] * (sumC[i] - sumC[j]) + s * (sumC[n] - sumC[j]);
-
- // 维护下凸包(删除队尾不满足凸性的点)
- while (l < r && check(q[r - 1], q[r], i)) r--;
- q[++r] = i; // 将i加入队尾
- }
- cout << dp[n] << endl;
- return 0;
- }
复制代码
### 关键点说明
1. **凸包维护**:
- 队列 `q` 存储决策下标,保证相邻点斜率递增(下凸包)。
- 插入新点 `i` 时,若队尾两点 `(a,b)` 与 `(b,i)` 的斜率不满足凸性(即 `check()` 返回 `true`),则删除 `b`。
2. **二分查找**:
- 在队列中二分找到第一个满足 `(dp[j+1]-dp[j])/(sumC[j+1]-sumC[j]) >= k_i` 的 `j`,此时 `j` 是最优决策点。
3. **复杂度**:`O(n log n)`,适用于 `n ≤ 3e5`。
> 提示:若 `t_i ≥ 0`(`k_i` 单调递增),可直接用单调队列将复杂度降至 `O(n)`。本题因 `t_i` 可能为负,故需二分。
以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。 |
|