LeetCode 笔记系列八 Longest Valid Parentheses [lich你又想多了]
题目:Given a string containing just the characters '(' and ')', find the length of the longest valid (well-formed) parentheses substring.
For "(()", the longest valid parentheses substring is "()", which has length = 2.
Another example is ")()())", where the longest valid parentheses substring is "()()", which has length = 4.
拿到题目,哎呀,这不是典型的动态规划嘛,然后刷刷开始coding。这道题确实可以用动态规划,但是复杂度就上去了,事实证明也没法通过大集合的测试。不过还是可以当一个DP的练习。如果你比较忙,直接看解法二吧。
用longestValid[i][j]表示从S串中的字符i到j的最长well-formed表达式的长度;isValid[i][j]表示从i到j是否是一个valid的表达式。
如何判断longestValid[i][j]的取值呢?有下面几种情况:
假定有一个maxlength变量。
1. 如果i='(' 并且 j=')',
a. 如果j = i+1,那这是一个valid的表达式;
b. 如果isValid[i+1][j-1]为真,那这也是一个valid表达式;
c. 对所有在i到j之间的k,如果isValid[i,k]&&isValid[k+1,j]为真,那么这是一个valid表达式,这三种情况maxlength都是i到j的距离;
d.其他情况,maxlength = longestValid[i][j-1]和 longestValid[i+1][j]比较大的那个。
2. 如果i='(' 并且 j='(', maxlength等于longestValid[i][j-1]。
3. 如果i=')' 并且 j=')', maxlength等于longestValid[i+1][j]。
4. 如果i=')‘ 并且 j='(', maxlength等于longestValid[i-1][j-1].
判断完成后,longestValid[i][j]赋值于maxlength。
下面是代码,是不是很复杂很想给博客君一巴掌。
解法一:
private static int longestValidParentheses(String s) {
// Start typing your Java solution below
// DO NOT write main() function
if(s.length() == 0) return 0;
boolean[][] isValid = new boolean[s.length()][];//isValid[i][j] is true when from i to j, this is a valid parentheses expression
int[][] longestValid = new int[s.length()][];//the longest length of valid expression between i and j
for(int i = 0; i < s.length();i++){
isValid[i] = new boolean[s.length()];
longestValid[i] = new int[s.length()];
}
for(int j = 1; j < s.length();j++){
for(int i = j - 1; i >=0;i--){
int left = s.charAt(i);
int right = s.charAt(j);
int maxLength = 0;
if(left == '(' && right ==')'){
if(i + 1 == j) {
isValid[i][j] = true;
if(maxLength < 2) maxLength = 2;
}else {
if(isValid[i+1][j-1]){
isValid[i][j] = true;
if(maxLength < j - i + 1) maxLength = j - i + 1;
}else if(isValid[i][i+1] && isValid[j-1][j] &&
(isValid[i + 2][j - 2] || j - i == 3)){
isValid[i][j]=true;
if(maxLength < j - i + 1) maxLength = j - i + 1;
}else {
maxLength = Math.max(longestValid[i][j-1], longestValid[i+1][j]);
for(int k = i + 1; k < j;k++){
if(isValid[i][k] && isValid[k + 1][j]) {
maxLength = j - i + 1;
isValid[i][j]=true;
break;
}
}
}
}
}else {
if(left == '(' && right == '('
&& longestValid[i][j-1] > maxLength) maxLength = longestValid[i][j-1];
if(left == ')' && right == ')'
&& longestValid[i+1][j] > maxLength)maxLength = longestValid[i+1][j];
if(left ==')' && right == '('
&& longestValid[i+1][j-1] > maxLength){
maxLength = longestValid[i+1][j-1];
}
}
longestValid[i][j] = maxLength;
}
}
return longestValid[0][s.length()-1];
}
plus,这是O(n3)的。。。。“该吃药了。。亲”。如期望的,该解法在大集合的时候超时。
解法二:来自leetcode讨论组的写法。本来人有O(n)的解法,被楼主活生生地卖弄成了O(n3)。楼主真想挖个坑把自己埋了!!!
大家首先看,这个解法里面的stack,不是用来存左右括号的。人是来存左括号的index。本来么,右括号也不用存。遍历S。遇到'(',放入lefts。如果遇到')',如果lefts是空,说明这是一个无法匹配的')',记录下last。last里面存放的其实是最后一个无法匹配的')'。为啥要保存这个值呢?主要是为了计算后面完整的表达式的长度。可以这样理解: “所有无法匹配的')'”的index其实都是各个group的分界点。
public static int longestValidParentheses2(String s) {
int maxLen = 0, last = -1;
Stack<Integer> lefts = new Stack<Integer>();
for (int i=0; i<s.length(); ++i) {
if (s.charAt(i)=='(') {
lefts.push(i);
} else {
if (lefts.isEmpty()) {
// no matching left
last = i;
} else {
// find a matching pair
lefts.pop();
if (lefts.isEmpty()) {//有一个完整的valid的group。计算该group的长度
maxLen = Math.max(maxLen, i-last);
} else {
//栈内还有‘(',一个最外层完整的group还没有匹配完成,
//但是通过查询下一个即将匹配还未匹配的"("的index来更新maxLen。
maxLen = Math.max(maxLen, i-lefts.peek());
}
}
}
}
return maxLen;
}
总结下:
DP不是万能的。注意发现问题的本质。不过这个确实要靠足够的练习。解法二的代码确实很简洁,但是并不是人人都能想到的。
LeetCode 笔记系列八 Longest Valid Parentheses [lich你又想多了]的更多相关文章
- leetcode第31题--Longest Valid Parentheses
Given a string containing just the characters '(' and ')', find the length of the longest valid (wel ...
- leetcode解题报告 32. Longest Valid Parentheses 用stack的解法
第一道被我AC的hard题!菜鸡难免激动一下,不要鄙视.. Given a string containing just the characters '(' and ')', find the le ...
- LeetCode (32) Longest Valid Parentheses
题目 Given a string containing just the characters '(' and ')', find the length of the longest valid ( ...
- leetcode解题报告 32. Longest Valid Parentheses 动态规划DP解
dp[i]表示以s[i]结尾的完全匹配的最大字符串的长度. dp[] = ; ; 开始递推 s[i] = ')' 的情况 先想到了两种情况: 1.s[i-1] = '(' 相邻匹配 这种情况下,dp ...
- 【leetcode刷题笔记】Longest Valid Parentheses
Given a string containing just the characters '(' and ')', find the length of the longest valid (wel ...
- 【一天一道LeetCode】#32. Longest Valid Parentheses
一天一道LeetCode系列 (一)题目 Given a string containing just the characters '(' and ')', find the length of t ...
- [LeetCode] Longest Valid Parentheses 最长有效括号
Given a string containing just the characters '(' and ')', find the length of the longest valid (wel ...
- Java for LeetCode 032 Longest Valid Parentheses
Given a string containing just the characters '(' and ')', find the length of the longest valid (wel ...
- [LeetCode] Longest Valid Parentheses 解题思路
Given a string containing just the characters '(' and ')', find the length of the longest valid (wel ...
随机推荐
- mac shell终端编辑命令行快捷键——行首行尾
mac shell终端编辑命令行快捷键——行首行尾 ctrl+a //移到行首ctrl+e //移到行尾===========linux系统用============alt+a //移到光标所在单词首 ...
- 动态时间规整-DTW算法
作者:桂. 时间:2017-05-31 16:17:29 链接:http://www.cnblogs.com/xingshansi/p/6924911.html 前言 动态时间规整(Dynamic ...
- ubuntu内窗口最大最小化
我用的dell笔记本,开始的时候,需要修改dell笔记本的BIOS,里面有一个 Funtion Key 一开始定义的行为是多媒体的,需要改成默认行为,具体记不清了, 反正知道这个,肯定知道哪里找. 然 ...
- linode使用apt更新时停止的错误
使用ubuntu在更新系统时候,有时候会出现这样的问题: 0% [Connecting to security.ubuntu.com (2001:67c:1560:8001::14)] 问了下客户,他 ...
- 0x00 译文:Windows桌面应用入门选择
当你想要创建一个新的桌面应用的时候,第一个你需要做的决定是使用Win32 和COM 还是使用.NET两个选项. C++ 和 Win32 使你“离奖牌更近”,并使你可以获得最佳的应用性能..NET 使你 ...
- 为什么 Windows API 使用 stdcall 调用约定?
作者:知乎用户链接:https://www.zhihu.com/question/31453641/answer/52001143来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注 ...
- PHP uxf framework 在模版中加入url标签
1. 确保不修改discuz代码: 2. 继承discuz template类,重载parse_template 方法:由于discuz在模版引擎这一块没有考虑扩展性,对标签的解析全部写在一个方法中, ...
- Android基础总结(七)BroadcastReceiver
广播(掌握) 广播的概念 现实:电台通过发送广播发布消息,买个收音机,就能收听 Android:系统在产生某个事件时发送广播,应用程序使用广播接收者接收这个广播,就知道系统产生了什么事件. Andro ...
- Oracle 错误:ORA-06413: Connection not open 解决办法
http://blog.csdn.net/neso520/article/details/6037411 ——————————————————————————————————————————————— ...
- android 全屏设置
更改styles.xml文件 <!-- 去掉标题栏 --> <style name="AppTheme" parent="Theme.AppCompat ...