zoukankan      html  css  js  c++  java
  • 算法提高 合并石子

    ---恢复内容开始---

     算法提高 合并石子  
    时间限制:2.0s   内存限制:256.0MB
        
    问题描述
      在一条直线上有n堆石子,每堆有一定的数量,每次可以将两堆相邻的石子合并,合并后放在两堆的中间位置,合并的费用为两堆石子的总数。求把所有石子合并成一堆的最小花费。
    输入格式
      输入第一行包含一个整数n,表示石子的堆数。
      接下来一行,包含n个整数,按顺序给出每堆石子的大小 。
    输出格式
      输出一个整数,表示合并的最小花费。
    样例输入
    5
    1 2 3 4 5
    样例输出
    33
    数据规模和约定
      1<=n<=1000, 每堆石子至少1颗,最多10000颗。
    #include <iostream>
    #include <cstdio>
    #include <vector>
    #include <queue>
    #include <map>
    #include <stack>
    #include <cstring>
    #include <algorithm>
    #include <cstdlib>
    #define FOR(i,x,n) for(long i=x;i<n;i++)
    #define ll long long int
    #define INF 0x3f3f3f3f3f3f3f3f
    #define MOD 1000000007
    #define MAX_N 50005
    
    using namespace std;
    
    int m[1005][1005];
    int a[1005];
    
    int main()
    {
        //freopen("input1.txt", "r", stdin);
        //freopen("data.out", "w", stdout);
        int N;
        ll sum=0;
        scanf("%d",&N);
        FOR(i,0,N){
            scanf("%d",&a[i]);
            m[i][i]=0;
            //sum+=a[i];
        }
        FOR(i,1,N){
            FOR(jj,0,N-i){
                int j=jj+i;
                    int minn=INF;
                    int s=0;
                    FOR(k,jj,j){
                        s+=a[k];
                        int t=m[jj][k]+m[k+1][j];
                        if(t<minn){
                            minn=t;
                        }
                    }
                    s+=a[j];
                    m[jj][j]=minn+s;
            }
        }
        printf("%d",m[0][N-1]);
        //fclose(stdin);
        //fclose(stdout);
        return 0;
    }

    ---恢复内容结束---

  • 相关阅读:
    bfs,队列
    Wannafly挑战赛22 A计数器(裴蜀定理 gcd)
    素数筛模板
    HDU
    HDU
    控制精度-----直接截取,不需四舍五入
    jstl下载与配置
    B. Treasure Hunt
    动态数组vector
    Manacher算法
  • 原文地址:https://www.cnblogs.com/TWS-YIFEI/p/6557084.html
Copyright © 2011-2022 走看看