题目链接:
https://vjudge.net/problem/HRBUST-1818
题意:
一条直线上摆放着一行共n堆的石子。现要将石子有序地合并成一堆。规定每次只能选相邻的两堆合并成新的一堆,并将新的一堆石子数记为该次合并的得分。请编辑计算出将n堆石子合并成一堆的最小得分和将n堆石子合并成一堆的最大得分。
Input
输入有多组测试数据。
每组第一行为n(n<=100),表示有n堆石子,。
二行为n个用空格隔开的整数,依次表示这n堆石子的石子数量ai(0<ai<=100)
Output
每组测试数据输出有一行。输出将n堆石子合并成一堆的最小得分和将n堆石子合并成一堆的最大得分。 中间用空格分开。
Sample Input
3
1 2 3
Sample Output
9 11
代码:
#include<bits/stdc++.h>
using namespace std;
const int maxn=104;
const int inf=0x3f3f3f3f;
int a[maxn];
int dp[maxn][maxn];
int s[maxn][maxn];
int dp2[maxn][maxn];
int n;
int main(){
while(scanf("%d",&n)!=EOF){
memset(dp,0,sizeof dp);
memset(dp2,0,sizeof dp2);
memset(s,0,sizeof s);
for(int i=1;i<=n;i++){
scanf("%d",a+i);
a[i]+=a[i-1];
}
dp[0][0]=1;
for(int i=1;i<=n;i++){
s[i][i]=i;
}
for(int len=1;len<n;len++){
for(int i=1;i<=n;i++){
int j=i+len;
dp[i][j]=inf;
for(int k=s[i][j-1];k<=s[i+1][j];k++){ //平行四边形优化
if(dp[i][j]>dp[i][k]+dp[k+1][j]+a[j]-a[i-1]){
s[i][j]=k;
dp[i][j]=dp[i][k]+dp[k+1][j]+a[j]-a[i-1];
}
}
}
}
for(int len=1;len<n;len++){
for(int i=1;i<=n;i++){
int j=i+len;
for(int k=i;k<j;k++){
dp2[i][j]=max(dp2[i][j],dp2[i][k]+dp2[k+1][j]+a[j]-a[i-1]);
}
}
}
printf("%d %d\n",dp[1][n],dp2[1][n]);
}
return 0;
}