标签:
//先贴代码:
#include<iostream>
#include<cstdio>
#include<string.h>
#include<cstring>
#include<string>
#include<algorithm>
#include<cmath>
using namespace std;
const int maxn = 100005;
const int maxh = 101;
const int inf = 0x7fffffff;
int h[maxn];
int n, c;
int dp[2][maxh];
int DP(){
int t = 0;
for(int i = h[1]; i < maxh; ++i){
dp[t][i] = pow(i - h[1],2);
}
for(int i = 2; i <= n; ++i){ //enumerate i
int mk[maxh],ak[maxh];
for(int k = 0; k < maxh; ++k){
dp[1-t][k] = inf;
mk[k] = dp[t][k] == inf ? inf:(dp[t][k] - c * k);
ak[k] = dp[t][k] == inf ? inf:(dp[t][k] + c * k);
}
int mmin = inf, amin = inf;
int minrec[maxh], addrec[maxh];
for(int j = 0; j < maxh; ++j){
minrec[j] = j == 0 ? 0 : (mk[j] < mk[minrec[j - 1]] ? j: minrec[j-1]);
addrec[maxh - j - 1] = (j == 0) ? (maxh - 1) : ((ak[maxh - j - 1] < ak[addrec[maxh - j]]) ? (maxh - j - 1): addrec[maxh - j]);
}
for(int j = h[i-1]; j < maxh; ++j){
mmin = mk[minrec[j]];
amin = ak[addrec[j]];
int sub = pow(j - h[i],2);
dp[1-t][j] = min(mmin + c * j + sub,amin - c * j + sub);
}
amin = ak[addrec[h[i-1]]];
for(int j = h[i]; j < h[i-1]; ++j){
dp[1-t][j] = min(dp[1-t][j],amin - c * j +((int)pow(j - h[i],2)));
}
for(int j = 0; j < h[i]; ++j){
dp[1-t][j] = inf;
}
t = 1 - t;
}
return 0;
}
int main(){
while(scanf("%d%d",&n, &c)!=EOF){
for(int i = 1; i <= n; ++i){
scanf("%d", h + i);
}
for(int i = 0; i < 2; ++i){
for(int j = 0; j < maxh; ++j){
dp[i][j] = inf;
}
}
DP();
int mincost = 0x7fffffff;
int choic = n & 1;
choic = 1 - choic;
for(int i = h[n]; i < maxh; ++i){
mincost = min(mincost, dp[choic][i]);
}
printf("%d\n", mincost);
}
return 0;
}
题意:给一个杆子的高度的数组h[1] ... h[n], 在相邻两个杆子(首尾的杆子不看作相邻)之间布线,单位高度差的费用为c;布线前可以选择将杆子的高度提高,提高x单位高度时,费用为x * x;
求在这n个杆子之间布线后的最小费用
思路:动态规划
dp[i][j] 表示第i根杆子的高度为j是,前i根杆子的最小费用和,那么状态转移方程为
dp[i][j] = min{dp[i-1][k] + c * |j -k| + pow(j - h[i],2)}
j,k的枚举范围是100,i有100000个, 复杂度太高,仔细观察可以发现
dp[i][j] = min{ dp[i-1][k] - c * k + c * j + pow(j - h[i],2) a[i-1]<= k <= j
dp[i-1][k] + c * k - c * j + pow(j - h[i],2) j <= k <= 100 }
这样,就可以分别用minrec和addrec两个数组纪录从0到j里使得dp[i-1][k]-c*k最小和从j到100里使得d[i-1][k] + c *k最小的k(这个纪录的时间是O(maxh)的,而不是对每个j 都枚举k)
优化:1. 对每个杆子,枚举高度时的上限可以是输入数组中的最大值而不是100
2. minrec和addrec直接纪录和j相关的最优值而不是产生最优值的下标
标签:
原文地址:http://my.oschina.net/u/1421373/blog/381005