洛谷P1880 [NOI1995]石子合并

题目描述

  在一个圆形操场的四周摆放 N N N 堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的 2 2 2 堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分。
试设计出一个算法,计算出将 N N N 堆石子合并成 1 1 1 堆的最小得分和最大得分。

输入格式

  数据的第 1 1 1 行是正整数 N N N,表示有 N N N 堆石子。
  第 2 2 2 行有 N N N 个整数,第 i i i 个整数 a i a_i ai 表示第 i i i 堆石子的个数。

输出格式

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

输入输出样例

输入 #1

4
4 5 9 4

输出 #1

43
54

说明/提示

1 ≤ N ≤ 100 1\leq N\leq 100 1N100 0 ≤ a i ≤ 20 0\leq a_i\leq 20 0ai20

分析

  由于石子摆在圆形操场的四周,因此 n n n 堆石子会摆放成一个环,于是不妨将 n n n 堆石子复制一份,使得 a i = a i + n a_i=a_{i+n} ai=ai+n
  定义 d p m i n [ i ] [ j ] dp_{min}[i][j] dpmin[i][j] 为合并区间 [ i , j ] [i,j] [i,j] 内石子获得的最小得分,有状态转移方程 d p m i n [ i ] [ j ] = min ⁡ i ≤ k < j ( d p m i n [ i ] [ k ] + d p m i n [ k + 1 ] [ j ] + c o u n t ( i , j ) ) dp_{min}[i][j]=\min\limits_{i\le k<j}(dp_{min}[i][k]+dp_{min}[k+1][j]+count(i,j)) dpmin[i][j]=ik<jmin(dpmin[i][k]+dpmin[k+1][j]+count(i,j))。同理,定义 d p m a x [ i ] [ j ] dp_{max}[i][j] dpmax[i][j] 为合并区间 [ i , j ] [i,j] [i,j] 内石子获得的最大得分,有状态转移方程 d p m a x [ i ] [ j ] = max ⁡ i ≤ k < j ( d p m a x [ i ] [ k ] + d p m a x [ k + 1 ] [ j ] + c o u n t ( i , j ) ) dp_{max}[i][j]=\max\limits_{i\le k<j}(dp_{max}[i][k]+dp_{max}[k+1][j]+count(i,j)) dpmax[i][j]=ik<jmax(dpmax[i][k]+dpmax[k+1][j]+count(i,j))。所求即为 min ⁡ 1 ≤ i ≤ n − 1 d p m i n [ i ] [ i + n ] \min\limits_{1\le i\le n-1}dp_{min}[i][i+n] 1in1mindpmin[i][i+n] max ⁡ 1 ≤ i ≤ n − 1 d p m a x [ i ] [ i + n ] \max\limits_{1\le i\le n-1} dp_{max}[i][i+n] 1in1maxdpmax[i][i+n]

代码

#include<algorithm>
#include<cstring>
#include<iostream>
#include<cstdio>
#define N 302
using namespace std;
int dp_min[N][N];
int dp_max[N][N];
int n;
int sum[N];
int a[N];
int main()
{
	//初始化
	memset(dp_min,0x3f,sizeof(dp_min));
	memset(dp_max,0,sizeof(dp_max));
	int i,j,k;
	cin>>n;
	for(i=1;i<=n;i++) scanf("%d",a+i);
	for(i=n+1;i<=2*n;i++) a[i]=a[i-n];
	//O(1)获得count(i,j)
	for(i=1;i<=2*n;i++) sum[i]=sum[i-1]+a[i];
	for(j=1;j<=2*n;j++)//右边界
	{
		for(i=j;i>=1;i--)//左边界
		{
			if(i==j) dp_min[i][j]=dp_max[i][j]=0;
			for(k=i;k<j;k++)//分界点
			{
				dp_min[i][j]=min(dp_min[i][k]+dp_min[k+1][j]+sum[j]-sum[i-1],dp_min[i][j]);
				dp_max[i][j]=max(dp_max[i][k]+dp_max[k+1][j]+sum[j]-sum[i-1],dp_max[i][j]);
			}
		}
	}
	int ans_min=0x3f3f3f3f,ans_max=-1;
	for(i=1;i<=n-1;i++)
	{
		ans_min=min(ans_min,dp_min[i][i+n-1]);
		ans_max=max(ans_max,dp_max[i][i+n-1]);
	}
	cout<<ans_min<<endl<<ans_max<<endl;
	return 0;
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值