石子归并问题求解
这真是一个悲伤的故事啊,这么经典的一个动态规划问题搞了这么久才稍微搞清楚了那么一小点。多么痛的领悟,你曾是我的全部。
下面是题目描述:
N堆石子摆成一条线。现要将石子有次序地合并成一堆。规定每次只能选相邻的2堆石子合并成新的一堆,并将新的一堆石子数记为该次合并的代价。计算将N堆石子合并成一堆的最小代价。
例如: 1 2 3 4,有不少合并方法
1 2 3 4 => 3 3 4(3) => 6 4(9) => 10(19)
1 2 3 4 => 1 5 4(5) => 1 9(14) => 10(24)
1 2 3 4 => 1 2 7(7) => 3 7(10) => 10(20)
括号里面为总代价可以看出,第一种方法的代价最低,现在给出n堆石子的数量,计算最小合并代价。
Input
第1行:N(2 <= N <= 100)
第2 - N + 1:N堆石子的数量(1 <= A[i] <= 10000)
Output
输出最小合并代价
Input示例
4
1
2
3
4
Output示例
19
大家看懂了没有呢?
题目是要求我们合并一些石子,并且要求我们合并的代价最小。
如何使合并的代价最小呢?这里使用了动态规划的思想。可素我怎么赶脚总会有一种贪心方法能实现呢?好吧,是不是我多虑了呢?我的贪心思路大概是每次找出相邻的代价最小的两个单位,然后弹出这两个单位,押入他们的和,用优先队列维护。不知道这个方法能不能行得通呢?!哈哈哈哈
下面给出经典解释:
include
using namespace std;
define M 101
define INF 0xffffff
int n,f[M][M],sum[M][M],stone[M];
int main()
{
int i,j,k,t;
cin>>n;
for(i=1;i<=n;i++)
cin>>stone[i];
for(i=1;i<=n;i++)
{
f[i][i]=0;
sum[i][i]=stone[i];
for(j=i+1;j<=n;j++)
sum[i][j]=sum[i][j-1]+stone[j];
}
for(int len=2;len<=n;len++)//合并长度
{
for(i=1;i<=n-len+1;i++)
{
j=i+len-1;
f[i][j]=INF;
for(k=i;k<=j-1;k++)
{
if(f[i][j]>f[i][k]+f[k+1][j]+sum[i][j])
f[i][j]=f[i][k]+f[k+1][j]+sum[i][j];
}
}
}
cout<