ybt1274 合并石子
时空限制 1000ms/64MB
【题目描述】
在一个操场上一排地摆放着N堆石子。现要将石子有次序地合并成一堆。规定每次只能选相邻的2堆石子合并成新的一堆,并将新的一堆石子数记为该次合并的得分。
计算出将N堆石子合并成一堆的最小得分。
【输入】
第一行为一个正整数N (2≤N≤100);
以下N行,每行一个正整数,小于10000,分别表示第i堆石子的个数(1≤i≤N)。
【输出】
一个正整数,即最小得分。
【输入样例】
7
13
7
8
16
21
4
18
【输出样例】
239
代码
#include<iostream>
#include<algorithm>
using namespace std;
const int N = 105;
int n,stone[N],sum[N],f[N][N];
int main(){
cin>>n;
for (int i=1; i<=n; i++){
cin>>stone[i];
sum[i] = sum[i-1]+stone[i]; //前缀和
}
fill(f[0],f[0]+N*N,0x3fffffff);
for (int i=1; i<=n; i++) f[i][i]=0;
for (int len=2; len<=n; len++) //合并石子堆数
for (int i=1; i+len-1<=n; i++){ //i合并起点
int j=i+len-1; //j合并终点
for (int k=i; k<j; k++) //k最后合并位置 f[i][j]从i开始连续合并j堆石子的最优值
f[i][j] = min(f[i][j],f[i][k]+f[k+1][j]+sum[j]-sum[i-1]);
}
cout<<f[1][n]<<endl;
return 0;
}