【CF5C Longest Regular Bracket Sequence】题解
题目链接
题目
This is yet another problem dealing with regular bracket sequences.
We should remind you that a bracket sequence is called regular, if by inserting ?+? and ?1? into it we can get a correct mathematical expression. For example, sequences ?(())()?, ?()? and ?(()(()))? are regular, while ?)(?, ?(()? and ?(()))(? are not.
You are given a string of ?(? and ?)? characters. You are to find its longest substring that is a regular bracket sequence. You are to find the number of such substrings as well.
思路
拿个栈来维护括号序列,最后剩下在栈里的就是不能合并的。
总结
这道题是道pj-的题,却生生折腾我一个月。
他其实涉及的是一个很经典的括号匹配问题。对于这类问题,这里是其中用栈解决的一大问题。
其实,不仅左括号可以塞进栈里,右括号也可以。
同时,运用栈里剩下不能匹配的,也能求最长匹配的。
Code
// Problem: CF5C Longest Regular Bracket Sequence
// Contest: Luogu
// URL: https://www.luogu.com.cn/problem/CF5C
// Memory Limit: 250 MB
// Time Limit: 2000 ms
//
// Powered by CP Editor (https://cpeditor.org)
#include
using namespace std;
//#define int long long
inline int read(){int x=0,f=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;
ch=getchar();}while(ch>='0'&&ch<='9'){x=(x<<1)+
(x<<3)+(ch^48);ch=getchar();}return x*f;}
//#define M
//#define mo
#define N 1000010
int n, m, i, j, k;
int zhan[N], top, t[N], ans;
char s[N];
signed main()
{
// freopen("tiaoshi.in", "r", stdin);
// freopen("tiaoshi.out", "w", stdout);
scanf("%s", s+1); n=strlen(s+1);
for(i=1; i<=n; ++i)
{
if(s[i]=='(')
zhan[++top]=i, t[top]=1;
if(s[i]==')')
{
if(top&&t[top]==1) --top;
else zhan[++top]=i, t[top]=2;
}
}
zhan[0]=0; zhan[++top]=n+1;
for(i=1; i<=top; ++i)
{
m=zhan[i]-zhan[i-1]-1;
// printf("%d\n", m);
if(m>ans) ans=m, k=0;
if(m==ans) ++k;
}
if(!ans) printf("0 1");
else printf("%d %d", ans, k);
return 0;
}