• 石子归并


    原题来自维基OI

    石子归并

    题目描述:有n堆石子排成一列,每堆石子有一个重量w[i],每次合并可以合并相邻的两堆石子,一次合并的代价为两堆石子的重量和w[i]+w[i+1].问安排怎样的合并顺序,能够使得总合并代价达到最小.

    输入描述:

    第一行一个整数n(n<=100)

    第二行n个整数w1,w2...wn(wi <= 100)

    输出描述:

    一个整数表示最小合并代价

    大概思路:

    求出w的前缀和s,s[0]设为0,s[i]表示w[1]+w[2]+...+w[i],这样的话w[i]+w[i+1]+...+w[j]就是s[j]-s[i-1].

    f[i][j]表示从第i堆合并到第j堆的最小代价,则f[i][j] = min(f[i][k]+f[k+1][j]+s[j]-s[i-1])(i<=k<=j),f[i][j]初始设置为INT_MAX.

    三层循环,最外层枚举一串合并的长度,最小是2堆石子合并,最大是n堆.

    然后枚举i,那么j就是i+len-1,在i和j之间枚举破点k,比较.

    AC代码如下:  

    #include <iostream>
    #include <algorithm>
    #include <climits>
    
    using namespace std;
    
    int main()
    {
        int n, s[105] = {0}, f[105][105], j;
        cin >> n;
        for (int i = 1; i <= n; i++)
        {
            cin >> s[i];
            s[i] += s[i-1];
        }
        for (int len = 2; len <= n; len++)
            for (int i = 1; i <= n - len + 1; i++)
            {
                j = i + len - 1;
                f[i][j] = INT_MAX;
                for (int k = i; k <= j - 1; k++)
                    f[i][j] = min(f[i][j], f[i][k] + f[k+1][j] + s[j] - s[i-1]);
            }
        cout << f[1][n];
        return 0;
    }

    如果有什么不合理之处,敬请批评指正.

  • 相关阅读:
    Java_zip_多源文件压缩到指定目录下
    Linux:ps -ef命令
    Linux:find命令中
    String.split()与StringUtils.split()的区别
    StringUtils工具类的常用方法
    Java:substring()
    Java:indexof()
    Linux:grep命令
    12、spring工厂+web前台搭建
    11、redis使用ruby实现集群高可用
  • 原文地址:https://www.cnblogs.com/zkx06111/p/3525455.html
Copyright © 2020-2023  润新知