RQNOJ 490 环形石子合并

题目链接:https://www.rqnoj.cn/problem/490

 

题目描述

在一个园形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分。

试设计出1个算法,计算出将N堆石子合并成1堆的最小得分和最大得分.

 

输入格式

数据的第1行试正整数N,1≤N≤100,表示有N堆石子.第2行有N个数,分别表示每堆石子的个数.

 

输出格式

输出共2行,第1行为最小得分,第2行为最大得分.

 

样例输入

4
4 4 5 9

样例输出

43
54

最开始没看到环形,就以为是普通的石子合并,交了就WA了,题意都没看懂。

好坑啊,不过思路跟普通的石子合并差不多,代码能看懂。

 

代码:

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
#define ll long long
const int maxn=205;
const int INF=0x3f3f3f3f;

int dp1[maxn][maxn],dp2[maxn][maxn];  ///dp1最小 dp2最大
int a[maxn],sum[maxn];
int n;

int main()
{
    //freopen("in.txt","r",stdin);
    while(scanf("%d",&n)==1)
    {
        for(int i=1; i<=n; i++)
        {
            scanf("%d",&a[i]);
            a[i+n]=a[i];
        }
        sum[0]=0;
        for(int i=1; i<=n*2; i++)
            sum[i]=sum[i-1]+a[i];
        memset(dp1,0,sizeof(dp1));
        memset(dp2,0,sizeof(dp2));

        for(int d=1; d<n*2; d++)
            for(int i=1; i+d<=n*2; i++)
            {
                int j=i+d;
                dp1[i][j]=INF;
                dp2[i][j]=-INF;
                for(int k=i; k<j; k++)
                {
                    dp1[i][j]=min(dp1[i][j],dp1[i][k]+dp1[k+1][j]+sum[j]-sum[i-1]);
                    dp2[i][j]=max(dp2[i][j],dp2[i][k]+dp2[k+1][j]+sum[j]-sum[i-1]);
                }
            }

        int ans1=INF;
        int ans2=-INF;
        for(int i=1; i<=n; i++)
        {
            ans1=min(ans1,dp1[i][i+n-1]);
            ans2=max(ans2,dp2[i][i+n-1]);
        }
        printf("%d\n",ans1);
        printf("%d\n",ans2);
    }
    return 0;
}

 

 

 

 

posted @ 2016-12-12 14:56  a_clown_cz  阅读(341)  评论(0)    收藏  举报