玩具装箱[HNOI2008]

题目描述

P 教授要去看奥运,但是他舍不下他的玩具,于是他决定把所有的玩具运到北京。他使用自己的压缩器进行压缩,其可以将任意物品变成一堆,再放到一种特殊的一维容器中。

P 教授有编号为 \(1 \cdots n\)\(n\) 件玩具,第 \(i\) 件玩具经过压缩后的一维长度为 \(C_i\)

为了方便整理,P教授要求:

  • 在一个一维容器中的玩具编号是连续的。
  • 同时如果一个一维容器中有多个玩具,那么两件玩具之间要加入一个单位长度的填充物。形式地说,如果将第 \(i\) 件玩具到第 \(j\) 个玩具放到一个容器中,那么容器的长度将为 \(x=j-i+\sum\limits_{k=i}^{j}C_k\)

制作容器的费用与容器的长度有关,根据教授研究,如果容器长度为 \(x\),其制作费用为 \((x-L)^2\)。其中 \(L\) 是一个常量。P 教授不关心容器的数目,他可以制作出任意长度的容器,甚至超过 \(L\)。但他希望所有容器的总费用最小。

输入格式

第一行有两个整数,用一个空格隔开,分别代表 \(n\)\(L\)

\(2\) 到 第 \((n + 1)\) 行,每行一个整数,第 \((i + 1)\) 行的整数代表第 \(i\) 件玩具的长度 \(C_i\)

输出格式

输出一行一个整数,代表所有容器的总费用最小是多少。

补一道斜率优化模板题QWQ

题解

\(dp[i]\)表示装前\(i\)个玩具的最小花费,\(sum[i]\)为前\(i\)个玩具的\(C_i\)之和

转移方程非常显然:\(dp[i]=\min_{j=0}^{i-1} dp[j]+(sum[i]-sum[j]+i-j-1-L)^2\)

一个经典的1D/1D动态规划

直接转移是\(n^2\)的 由于这题转移方程有平方项所以也不能直接用单调队列 看到有二次项的1D/1D动态规划就要考虑使用斜率优化

接下来说说什么是斜率优化

我们给原方程变个形

\(A_i=sum[i]+i,B_i=sum[i]+i+L+1\)

那个\(\min\)看着麻烦 先扔掉了

那么现在有 \(dp[i]=dp[j]+(A_i-B_j)^2\)

拆开完全平方,得到\(dp[i]=dp[j]+A_i^2-2*A_i*B_j+B_j^2\)

把只含\(j\)的全部移到左边,得到\((dp[j]+B_j^2)=(2*A_i)*B_j+(dp[i]-A_i^2)\)

为什么要打上括号?

看一下直线的斜截式的方程\(y=kx+b\) 我们的转移方程里 让\(y=dp[j]+B_j^2, k=(2*A_i), x=B_j, b=(dp[i]-A_i^2)\) 就是一条直线方程

其中\(j<i\),所以\(dp[j]\)我们已经算出来了 所以\(y,k,x\)都是已知的

\(j<i\)\(j\)不止一个 所以实际上现在在二维平面上有很多个点 第\(j\)个点的坐标为\((B_j, dp[j]+B_j^2)\)

由于\(A_i^2\)已知,我们希望\(dp[i]\)尽量小就是要让\(dp[i]-A_i^2\)尽量小 也就是截距要尽量小

所以我们现在需要在二维平面上选一个点\(j\) 过它画一条斜率为\(2*A_i\)的直线 直线的截距即是\(dp[i]-A_i^2\)

比如像是这样?

这是我们在推\(dp[6]\) 当直线斜率为\(2*A_6\)时 我们发现让它经过第3个点得到的截距最小

从而我们就能让\(dp[6]\)的值从\(dp[3]\)转移过来 然后在图中加入第6个点\((B_6,dp[6]+B_6^2)\)

问题来了 我们怎么快速求得经过哪个点会使得截距最小?

这题有一个很重要的性质:\(x\),即\(B_j\); 以及\(k\),即\(2*A_i\)(满足决策单调性!!!); 这两个都是递增的 (没有这个性质就不能用单调队列了 而是用cdq分治之类的算法)

所以第\(i\)个点一定在第\(i-1\)个点右边 而且直线的斜率也只会越来越大

我们对于已经存在的点 维护一个下凸壳 放在单调队列里

我们记第\(j\)个点为\(p_j\),记相邻两点连成的直线的斜率为 \(\operatorname{slope}(p_{j-1},p_j)\)

假设当前在推\(dp[6]\)的值 则我们要画一条斜率为\(2*A_6\)的直线

观察一下 我们发现 找到凸壳上第一个\(j\)满足\(\operatorname{slope}(p_{j},p_{j+1})>2*A_6\),过这个点画直线就是最优的 这个例子中这个点就是3

而且还有一个之前提到的性质 \(2*A_i\)是单增的 所以如果某个\(\operatorname{slope}(p_{j},p_{j+1})<2*A_6\) 那么对于之后的\(i>6\),也一定是\(\operatorname{slope}(p_{j},p_{j+1})<2*A_i\) 所以\(j\)这个点无论如何都不再会被用到 就直接从队列里弹出

最后单调队列的队头就是最优的决策点 我们算出\(dp[6]\)的值 由于\(B_j\)是递增的 我们可以直接把\((B_6,dp[6]+B_6^2)\)拿去更新这个凸包

更新时\((B_6,dp[6]+B_6^2)\)可能在这里

此时 \(\operatorname{slope}(p_{4},p_{5})>\operatorname{slope}(p_{5},p_{6})\)

为了维护凸包的性质 我们把\(4\rightarrow 5\)这条边从队列里删掉(即把5弹出) 然后再加入6

实现起来其实很简单 结合注释看看吧

每个点最多出队入队一次 时间复杂度\(O(n)\)
不知道为什么原题数据规模这么小

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;

template<typename T>
inline void read(T &num) {
	T x = 0, f = 1; char ch = getchar();
	for (; ch > '9' || ch < '0'; ch = getchar()) if (ch == '-') f = -1;
	for (; ch <= '9' && ch >= '0'; ch = getchar()) x = (x << 3) + (x << 1) + (ch ^ '0');
	num = x * f;
}

int n, head, tail, q[100005]; 
ll l, c[100005], a[100005], b[100005], dp[100005];

struct point{
	ll x, y;
	point() {}
	point(ll xx, ll yy): x(xx), y(yy) {}
} p[100005];

double slope(point A, point B) {
	return 1.0 * (A.y - B.y) / (A.x - B.x);
}

int main() {
	read(n); read(l);
	b[0] = l + 1;
	for (int i = 1; i <= n; i++) {
		read(c[i]); 
		a[i] = a[i-1] + c[i] + 1; //预处理Ai,Bi
		b[i] = b[i-1] + c[i] + 1;
	}
	head = tail = 1;
	p[0] = point(b[0], b[0] * b[0]); //开始时队列里只有0号点
	q[1] = 0;
	for (int i = 1; i <= n; i++) {
		while (head < tail && slope(p[q[head]], p[q[head+1]]) < 2.0 * a[i] + 1e-8) head++; //文中注释1
		dp[i] = dp[q[head]] + (a[i] - b[q[head]]) * (a[i] - b[q[head]]); //q[head]为最佳决策点 直接用原转移方程更新dp[i]
		p[i] = point(b[i], dp[i] + b[i] * b[i]); //确定i号点的位置
		while (head < tail && slope(p[q[tail-1]], p[q[tail]]) + 1e-8 > slope(p[q[tail]], p[i])) tail--; //文中注释2
		q[++tail] = i;
	}
	printf("%lld\n", dp[n]);
	return 0;
}
posted @ 2020-05-10 21:52  AK_DREAM  阅读(106)  评论(0编辑  收藏  举报