【LG P1944】最长括号匹配

时间:2021-07-19 16:39:33   收藏:0   阅读:0

算法方向分析

对于这题 \(\le 1000000\) 的数据规模显然只允许我们用一重循环。

最长,可见这是道最值问题

最值问题可以用贪心,DP,二分……

这道题我们用 DP 来做。

构建状态

首先,我们需要构建状态,状态的构建不是唯一的,受最长上升子序列的影响,我是这样构建的:令 \(f_i\) 为以 \(s_i\) 结尾的字符串最长匹配。

考虑 \(s_i\),要想它能构成括号匹配,很显然地,它肯定不能为(或者[

我们要找到一个 \(s_k=\) ([,使得在 \((k,i)\)这个开区间内的字符串为最长括号匹配且 \([k,i]\) 这个闭区间尽可能得大。

那么什么情况能满足上面的条件呢?

考虑每一个 \(f_{i-1}\),它是以 \(s_{i-1}\) 结尾的最长括号匹配。当 \(s_i\)\(\large s_{i-1-f_{i-1}}\) 匹配时,长度增加 \(2\),链接起前面的最长括号匹配,如图所示。

技术图片

因此公式为:

\[\huge f_i=f_{i-1}+2+f_{i-f_{i-1}-2} \]

如果难理解的话,不如看这个栗子 ((()[]())(每个字符记为 \(s_{1,2,\dots,6}\)),带入公式得:

\[\large f_9=f_8+2+f_{6} \]

对于第 \(9\) 个字符,它可以与它对应的括号共同对答案贡献 \(2\) 个。但这个区间里不一定只有一个镶嵌区间,它可能有多个。因此我们要把上一最长区间加起来。上一组区间计算时已经把上上一组区间包含在内,因此无需重复计算。

那么若 \(\large s_{i-1-f_{i-1}}\)\(s_i\) 不匹配怎么办?这时 \(f_i\) 肯定为 \(0\),因为除此之外,不管选哪个字符,都没法满足它和 \(s_i\) 构成括号匹配。

大家自己画个图动手算一算就很明显了。

分析千万条,代码第一条,不懂的看一下代码。

代码

#include<bits/stdc++.h>
using namespace std;
const int N=1e6+5;
int f[N];
int main() {
	string s;
	cin>>s;
	int ans=0,id;
	for(int i=1; i<s.size(); i++) {
		if(s[i]==‘(‘||s[i]==‘[‘) {
			continue;
		}
		if((s[i]==‘)‘&&s[i-f[i-1]-1]==‘(‘)||(s[i]==‘]‘&&s[i-f[i-1]-1]==‘[‘)) {
			f[i]=f[i-1]+2+f[i-f[i-1]-2];
			if(f[i]>ans) {
				ans=f[i],id=i;
			}
		}
	}
	for(int i=id-ans+1; i<=id; ++i) {
		printf("%c",s[i]);
	}
	return 0;
}

参考资料

题解 P1944 【最长括号匹配】 - OItby - 洛谷博客

评论(0
© 2014 mamicode.com 版权所有 京ICP备13008772号-2  联系我们:gaon5@hotmail.com
迷上了代码!