区间dp-环形石子合并

文章描述了如何使用环形结构、前缀和和memset函数在C++中解决一个问题,涉及到区间和的计算以及更新最小最大值的操作。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

传送门icon-default.png?t=N7T8https://www.luogu.com.cn/problem/P1880

相对于弱化版     这里需要注意(不只是因为环形)

  1. 1.环形做法(环成链)
  2. 2.前缀和维护(不太好计算区间和)
  3. 3.memset需要用cstring头文件
  4. 4.对f【0】【0】的操作

文字无力,见代码

#include<iostream>
#include<cstring>//别忘了,因为这个WA一发
using namespace std;
const int N=305;
int f[N][N];
int g[N][N];//分别记录大小数组
int s[N];//前缀和
int a[N];
int minn=1e9;
int maxn=-1e9;
#define inf 0x3f

int main(){
	memset(f,0x3f,sizeof f);//先全部设为最大,这样方便后面操作
	memset(g,0,sizeof g);//不会是负数,0即可
	int n;cin>>n;
	for(int i=1;i<=n;++i){
		cin>>a[i];a[i+n]=a[i];//环成链
	}
	for(int i=1;i<=2*n;++i){
		s[i]=s[i-1]+a[i];f[i][i]=0,g[i][i]=0;//前缀和和前置操作  (f【0】【0】)
	}
	for(int len=2;len<=n;++len){//长度
		for(int i=1;(i+len-1)<=2*n;++i){//起点
			int j=i+len-1;//终点
			for(int k=i;k<j;++k){//从哪里开始分开
				f[i][j]=min(f[i][j],f[i][k]+f[k+1][j]+s[j]-s[i-1]);
				g[i][j]=max(g[i][j],g[i][k]+g[k+1][j]+s[j]-s[i-1]);
                
                minn=min(minn,f[i][i+n-1]);//这里如果没有被操作,将会一直是最大值
                maxn=max(maxn,g[i][i+n-1]);
			}
		}
	}
	cout<<minn<<endl<<maxn;
	
	return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值