luogu 2308添加括号

添加括号

传送门

题目大意

现在要添上n-1对括号,加法运算依括号顺序进行,得到n-1个中间和,求出使中间和之和最小的添括号方法。

这道题其实是一个很简单的区间dp,中间和的意思是括号里面的和,也就是说,一个括号就有一个中间和,然后求总的中间和。

设dp[l][r]表示区间\([l,r]\)内最大中间和是多少,然后dp方程也是一个很简单的入门级方程
\[dp[l][r]=min(dp[l][r],dp[l][k]+dp[k+1][r]+sum[r]-sum[l-1])\]
枚举到一个区间,表示把这个区间两端加上括号。

然后到了这道题的关键部分,怎么输出在那个地方添加括号以及每一个中间和

我们一步一步来说
首先我们用到一个断点记录数组,记录区间\([l,r]\)的最优值断点处

  1. 输出括号添加的序列,辅助数组lc[],rc[],然后递归改变两个辅助数组的值,然后输出括号序列
  2. 输出每一部分的中间和,因为由小到大,所以也是递归输出。
#include <iostream>
#include <algorithm>
#include <queue>
#include <cstdio>
#include <cstring>
using namespace std;
int lc[50],rc[50],sum[50],n,a[50],dp[50][50],cirl[50][50];
void Print(int l,int r) {
    if(l==r)return;
    ++lc[l];
    ++rc[r];
    Print(l,cirl[l][r]);
    Print(cirl[l][r]+1,r);
}
void Prinf(int l,int r) {
    if(l==r)return ;
    Prinf(l,cirl[l][r]);
    Prinf(cirl[l][r]+1,r);
    cout << sum[r]-sum[l-1]<<' ';
}
int main() {
    scanf("%d",&n);
    memset(dp,127/3,sizeof(dp));
    for(int i=1; i<=n; i++) {
        scanf("%d",&a[i]);
        dp[i][i]=0;
        sum[i]=sum[i-1]+a[i];
    }
    for(int len=2; len<=n; len++)
        for(int l=1,r=len+l-1; r<=n; l++,r++)
            for(int k=l; k<=r; k++)
                if(dp[l][r]>=dp[l][k]+dp[k+1][r]+sum[r]-sum[l-1]) {
                    dp[l][r]=dp[l][k]+dp[k+1][r]+sum[r]-sum[l-1];
                    cirl[l][r]=k;
                }
    Print(1,n);
    for(int i=1; i<=n; i++) {
        for(int j=1; j<=lc[i]; j++)
            cout << "(";
        cout << a[i];
        for(int j=1; j<=rc[i]; j++)
            cout << ")";
        if(i!=n)cout << '+';
    }
    cout << endl;
    cout << dp[1][n]<<endl;
    Prinf(1,n);
        return 0;
}

转载于:https://www.cnblogs.com/ifmyt/p/9660922.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值