HDU 3507 Print Article (斜率优化)

HDU 3507 Print Article (斜率优化)

ACM

题目地址: 
HDU 3507 Print Article

题意: 
给定一个长度为n的序列。和一个常数m,我们能够将序列分成任意段,每段的权值为sum(arr[i]) + C(x<=i<=y),求一种划分方法使得整个序列的权值最小

分析: 
from:亟隐's blog

f[i]=min(f[k]+(sum(i)-sum(k))^2 )+m 
= f[k]+sum^2(i)+sum^2(k)-2*sum(i)*sum(k)+m. 
也就是找一个最小的f[k]+sum^2(k)-2*sum(i)*sum(k),假设k优于j,必有(f[k]+sum^2(k)-f[j]-sum^2(j))/(2*sum(k)-2*sum(j))<=s[i].

代码

/*
*  Author:      illuz <iilluzen[at]gmail.com>
*  File:        3507.cpp
*  Create Date: 2014-09-19 10:19:57
*  Descripton:  dp
*/

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;

#define repf(i,a,b) for(int i=(a);i<=(b);i++)
typedef long long ll;

const int N = 5e6 + 10;

int n, m, l, r, q[N];
ll s[N], x[N], y[N], k[N];

void init() {
	int x;
	repf (i, 1, n) {
		cin >> x;
		s[i] = s[i - 1] + x;
		k[i] = s[i] * 2;
	}
}

inline bool check(int a, int b, int c) {
	return (y[a] - y[b]) * (k[b] - k[c]) <= (y[b] - y[c]) * (k[a] - k[b]);
}

void solve() {
	q[0] = l = r = 0;
	repf (i, 1, n) {
		while (l < r && (y[q[l+1]]-y[q[l]]) <= s[i] * (k[q[l+1]]-k[q[l]]))
			l++;
		x[i] = x[q[l]] + (s[i] - s[q[l]]) * (s[i] - s[q[l]]) + m;
		y[i] = x[i] + s[i] * s[i];
		while (l < r && check(i, q[r], q[r-1]))
			r--;
		q[++r] = i;
	}
	cout << x[n] << endl;
}

int main() {
	ios_base::sync_with_stdio(0);
	while (cin >> n >> m) {
		init();
		solve();
	}
	return 0;
}


posted @ 2017-06-03 11:17  wzzkaifa  阅读(128)  评论(0编辑  收藏  举报