洛谷P2308 添加括号【区间DP+记录合并方案】【绿】

博客探讨了洛谷P2308题目,即如何在不改变元素位置的情况下,通过添括号使得中间和之和最小。文章介绍了问题背景、输入输出格式和样例,并提供了动态规划的解题思路,强调了在递归过程中记录分界点的重要性。代码实现部分展示了如何通过递归求解并构造正确的括号序列。

Date:2022.03.25
题目背景
给定一个正整数序列a(1),a(2),…,a(n),(1<=n<=20)
不改变序列中每个元素在序列中的位置,把它们相加,并用括号记每次加法所得的和,称为中间和。
例如:
给出序列是4,1,2,3。
第一种添括号方法:
((4+1)+(2+3))=((5)+(5))=(10)
有三个中间和是5,5,10,它们之和为:5+5+10=20
第二种添括号方法
(4+((1+2)+3))=(4+((3)+3))=(4+(6))=(10)
中间和是3,6,10,它们之和为19。
题目描述
现在要添上n-1对括号,加法运算依括号顺序进行,得到n-1个中间和,求出使中间和之和最小的添括号方法。
输入格式
共两行。 第一行,为整数n。(1< =n< =20) 第二行,为a(1),a(2),…,a(n)这n个正整数,每个数字不超过100。
输出格式
输出3行。 第一行,为添加括号的方法。 第二行,为最终的中间和之和。 第三行,为n-1个中间和,按照从里到外,从左到右的顺序输出。
输入输出样例
输入 #1复制
4
4 1 2 3
输出 #1复制
(4+((1+2)+3))
19
3 6 10

思路:石子合并过程即为一个二叉树递归,其核心在于找到每一步的分界点,因此第一次区间dp求出最优解同时记录下这个区间的分界点。第三行要递归得出,但需要注意这个过程要保证深层的、先合并的先输出,因此体现在代码上是先递归再输出。第一行不会看了题解,发现也是递归,每次区间是[l,r][l,r][l,r],因此向下递归即在lll左边加上′(′'('(rrr右边加上′)′')'),最后根据①左括号②数③右括号④加号输出,一定是对的。
代码如下:

#include <bits/stdc++.h>
#define x first
#define y second
using namespace std;
typedef long long LL;
typedef pair<LL, LL> PII;
const LL N = 550,INF=0x3f3f3f3f3f3f3f3f;
typedef long long LL;
LL t,n,m,a[N],f[N][N],sum[N],path[N][N],zuo[N],you[N];
bool st[N];
set<LL>s;
void print(LL l,LL r)
{
    if(l==r) return;
    LL k=path[l][r];
    if(k!=-1&&path[l][k]!=-1) print(l,k);
    if(k!=-1&&path[k+1][r]!=-1) print(k+1,r);
    cout<<sum[r]-sum[l-1]<<' ';//先递归,再输出
}
void kuohao(LL l,LL r)
{
    if(l==r) return;
    LL k=path[l][r];
    zuo[l]++;you[r]++;
    if(k!=-1&&path[l][k]!=-1) kuohao(l,k);
    if(k!=-1&&path[k+1][r]!=-1) kuohao(k+1,r);
}
int main()
{
    ios::sync_with_stdio(0);cin.tie(0);cout.tie(0);
    cin>>n;
    for(int i=1;i<=n;i++) {cin>>a[i];sum[i]=sum[i-1]+a[i];path[i][i]=-1;}
    for(int len=2;len<=n;len++)
        for(int l=1;l+len-1<=n;l++)
        {
            LL r=l+len-1;f[l][r]=1e18;
            for(int k=l;k<r;k++)
                if(f[l][k]+f[k+1][r]+(sum[r]-sum[l-1])<f[l][r])
                {
                    f[l][r]=f[l][k]+f[k+1][r]+(sum[r]-sum[l-1]);
                    path[l][r]=k;
                }
        }
    kuohao(1,n);
    for(int i=1;i<=n;i++)
    {
        for(int j=1;j<=zuo[i];j++) cout<<"(";
        cout<<a[i];
        for(int j=1;j<=you[i];j++) cout<<")";
        if(i!=n) cout<<"+";
    }
    cout<<'\n';
    cout<<f[1][n]<<'\n';
    print(1,n);
    return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值