斜率优化初探:以 [HNOI2008]玩具装箱 为例
记
直接转移是
斜率优化的过程
(一)问题转化成了求最小截距。
我们把
把括号打开,可以得到:
移项后得到:
此时,如果我们把这看做一个一次函数,那么
注意到固定
(二)截距在哪里最小?(图像理解)
我们知道有用的
我们惊讶的发现斜率竟然是固定的!我们可以平移这条直线至与凸包相切,显然这个切点
怎么找这个点呢?发现
(三)决策单调性的优化(图像理解)
决策单调性的定义:
设
发现
详细证明参见参考博客。
用单调队列维护凸包的点集,分三步:
- 将斜率比目标斜率小的点弹出, 在队首位置找到最优决策点
。 - 用最优决策点
更新 。 - 把新的点加入队列中。
时间复杂度
#include<bits/stdc++.h>
#define F(i,l,r) for(int i(l);i<=(r);++i)
#define G(i,r,l) for(int i(r);i>=(l);--i)
#define int ll
using namespace std;
using ll = long long;
const int N = 1e5 + 5;
int L, n, h = 1, t = 0;
int f[N], s[N], q[N];
int X(int j){
return s[j] +L;
}
int Y(int j){
return f[j] + (s[j] + L) * (s[j] + L);
}
long double slope(int i, int j){
return (long double)(Y(i) - Y(j)) / (long double)(X(i) - X(j));
}
signed main(){
ios::sync_with_stdio(0); cin.tie(0); cout.tie(0);
cin >> n >> L;
++L;
F(i, 1, n) cin >> s[i], s[i] += s[i - 1] + 1;
q[++t] = 0;
F(i, 1, n){
while(h < t && slope(q[h], q[h + 1]) < 2 * s[i]) ++ h;
int j = q[h];
f[i] = f[j] + (s[i] - s[j] - L) * (s[i] - s[j] - L);
while(h < t && slope(q[t - 1], q[t]) > slope(q[t - 1], i)) -- t;
q[++ t] = i;
}
cout << f[n] << '\n';
return 0;
}
反思:移项的依据
为了用
注意
参考博客:
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· AI技术革命,工作效率10个最佳AI工具
2023-10-08 题解:洛谷P1119 灾后重建