Skip to content
字数
597 字
阅读时间
3 分钟

典型例题:

石子合并

题目描述

设有 N 堆石子排成一排,其编号为1,2,3,..., N。 每堆石子有一定的质量,可以用一个整数来描述,现在要将这 N 堆石子合并成为一堆。

每次只能合并相邻的两堆,合并的代价为这两堆石子的质量之和,合并后与这两堆石子相邻的石子将和新堆相邻,合并时由于选择的顺序不同,合并的总代价也不相同。 例如有 4 堆石子分别为 1,3, 5, 2, 我们可以先合并第 1、 2 堆,代价为 4,得到 4 ,5 ,2, 又合并第 1、 2 堆,代价为 9, 得到 9, 2 ,再合并得到 11,总代价为 4 + 9 +11 = 24

如果第二步是先合并第 2、 3堆,则代价为 7,得到 4 ,7,最后一次合并代价为 11,总代价为 4 + 7 + 11 = 22

问题是:找出一种合理的方法,使总的代价最小,输出最小代价。

输入格式

第一行一个数 N 表示石子的堆数 第二行 N 个数,表示每堆石子的质量(均不超过 1000)

输出格式

输出一个整数,表示最小代价

数据范围

1≤N≤300

样例
输入1
4
1 3 5 2
输出1
22
题解
思路
mermaid
flowchart LR
A[动态规划] --> B["状态表示f[i,j]"]
A[动态规划] --> C[状态计算]
B["状态表示f[i,j]"] --> D["集合:所有将第i堆石子到第j堆石子合并成一堆石子的合并方式"]
B["状态表示f[i,j]"] --> E["属性:Min"]
代码
c++
// 区间DP
#include <iostream>
#include <algorithm>

using namespace std;

const int N = 1010;
int f[N][N];
int s[N];
int n;

int main()
{
    cin >> n;

    for (int i = 1; i <= n; i ++)   cin >> s[i];

    for (int i = 1; i <= n; i ++)   s[i] += s[i - 1];

    for (int len = 2; len <= n; len ++) // 区间长度
        for (int i = 1; i + len - 1 <= n; i ++) // 区间头和区间尾
        {
            int l = i, r = i + len - 1;
            f[l][r] = 1e8; // 因为要取最小,所以不能用默认0
            for (int k = l; k < r; k ++)
                f[l][r] = min(f[l][r],f[l][k] + f[k + 1][r] + s[r] - s[l - 1]);
        }
        
    cout << f[1][n];

    return 0;
}

贡献者

The avatar of contributor named as freeway348 freeway348

文件历史

撰写