牛客网NOIP赛前集训营-普及组(第一场)

C 括号

题目大意:对于输入的一串括号“()))(()()((”,可以删去若干位(删除位置不完全相同时均算作两种操作),若想最终得到合法括号串,求共有多少种操作方法可选。

先说算法后解释: 

dp[i][j]表示,加入第i个元素后,能将前i个字符转化为有j个未配对左括号‘(’的方法数。                                                                      以此为基础

dp[i][j] 的构成分为两部分,可根据操作中是否删去新字符来分类:                                                                                            不删,沿用dp[i-1][j]的方法即可;删去,则沿用dp[i-1][j-1]或dp[i-1][j+1]

对于0<j<n

若s[i]=='(' ,则dp[i][j]=dp[i-1][j]+dp[i-1][j-1]

若s[i]==')' ,则dp[i][j]=dp[i-1][j]+dp[i-1][j+1]

据此,每加入新字符,均对dp数组进行一次O(n)的更新,总复杂度O(n^2)

可进行空间优化,注意一下遍历的起始位置即可。附代码:

#include<iostream>
#include<cstdio>
#define ll long long

using namespace std;

const int mod=1e9+7;

int dp[10000+4];
char s[10000+4];
int n;

int main()
{
    cin>>n;
    scanf("%s",s+1);

    dp[0]=1;
    for(int i=1; i<=n; i++)
    {
        if(s[i]=='(')
        {
            for(int j=i-1; j>=0; j--)
                dp[j+1]=(dp[j+1]+dp[j])%mod;
        }
        else
        {
            for(int j=1; j<=i; j++)
                dp[j-1]=(dp[j-1]+dp[j])%mod;
        }
    }
    cout<<(dp[0]-1+mod)%mod<<endl;
    return 0;
}

思路:暴力枚举肯定无法应对(n==1000)的情况,考虑优化时努力向dp上靠拢。我们可以先观察对于一个子串(1~i),在其后加上一个括号‘(’或‘)’时,与旧字符串有什么关系。

 

全部评论

相关推荐

点赞 评论 收藏
分享
秋招进行到现在终于能写总结了。完全没想到战线会拉这么长,过程会如此狼狈,不过更应该怪自己太菜了。好在所有的运气都用在了最后,也是有个去处。背景:双2本硕科班,无竞赛,本科一段研究所实习,硕士一段大厂暑期实习但无转正。技术栈是C++&nbsp;&amp;&nbsp;Golang,实习是客户端音视频(而且是鸿蒙端开发),简历两个C++项目一个Golang项目。主要投递岗位:后端,cpp软开,游戏服务端,测开,以及一些不拘泥于Java的岗位。从8月起总共投递123家公司,笔试数不清了,约面大约30家。offer/oc/意向:友塔游戏(第一个offer,面试体验很好,就是给钱好少南瑞继保(计算机科班点击就送(限男生),不...
乡土丁真真:佬很厉害,羡慕~虽然我还没有到校招的时候,也想讲一下自己的看法:我觉得不是CPP的问题,佬的背书双2,技术栈加了GO,有两段实习。投了123,面了30.拿到11个offer。这个数据已经很耀眼了。这不也是CPP带来的吗?当然也不止是CPP。至少来说在这个方向努力过的也会有好的结果和选择。同等学历和项目选java就会有更好的吗?我个人持疑问态度。当然CPP在方向选择上确实让人头大,但是我觉得能上岸,至于最后做什么方向,在我看来并不重要。至于CPP特殊,有岗位方向的随机性,java不是不挑方向,只是没得选而已。也希望自己以后校招的时候能offer满满
点赞 评论 收藏
分享
评论
点赞
收藏
分享
牛客网
牛客企业服务