问题描述:
给定一排数,从中按顺序取出n-2个数,并且每取一次就把该数和相邻的数相乘,然后所有的n-2个乘积相加,要求和最小。
解题思路:
考虑区间[start, end], 如果在其中取出第j个数(start+1 <= j <= end - 1),那么在这之前j左边的和右边的数必然是分开的两个子问题,没有相交,于是可以得到最优子结构的性质。由于是区间,可以采用记忆化搜索。
dp[i][j] = min{dp[i][k] + dp[k][j] + a[i]*a[k]*a[j], i+1 <= k <= j-1}
代码如下:
#include <iostream>
using namespace std;
int dp[101][101];
int num[101];
int n;
int dfs(int s, int e)
{
if(e - s < 2)
return 0;
int i;
int Min = 1000000001;
for(i = s + 1; i <= e - 1; i++)
{
if(dp[s][i] == -1)
dp[s][i] = dfs(s, i);
if(dp[i][e] == -1)
dp[i][e] = dfs(i, e);
if(dp[s][i] + dp[i][e] + num[s] * num[i] * num[e] < Min){
Min = dp[s][i] + dp[i][e] + num[s] * num[i] * num[e];
}
}
return Min;
}
int main()
{
int i;
while(scanf("%d", &n) != EOF)
{
memset(dp, -1, sizeof(dp));
for(i = 1; i <= n; i++){
scanf("%d", &num[i]);
}
printf("%d\n", dfs(1, n));
}
}