美文网首页
32. 最长有效括号

32. 最长有效括号

作者: 一直流浪 | 来源:发表于2022-10-20 09:40 被阅读0次

32. 最长有效括号

题目链接:https://leetcode-cn.com/problems/longest-valid-parentheses/

难度:困难

给定一个只包含 '('')' 的字符串,找出最长的包含有效括号的子串的长度。

示例 1:

输入: "(()"
输出: 2
解释: 最长有效括号子串为 "()"

示例 2:

输入: ")()())"
输出: 4
解释: 最长有效括号子串为 "()()"

解法一:暴力法

根据题目意思,我们需要找到最长的包含有效括号的子串的长度,那么意味着,我们的最长子串不一定是原字符串的首元素开始的。

但是如果我们的最长子串都是从首元素开始的,那么这道题就会变得非常简单。我们只需要利用一个循环,挨个从0,1,……为首元素,开始截取字符串,然后在判断这个子串的最长有效长度,最后拿出最大值即可。

如何判断计算一个字符串从首元素开始的最长有效长度?

我们可以利用栈的思想,但是没必要使用栈数据结构,利用一个int变量temp(初始值为0)就可以代替:

(1)我们遍历字符串,若遇到 ' ( ' 就给temp++;若遇到 ' ) ' 就给temp--;

(2)若遇到 temp = 0,说明前面的字符串刚好满足条件,我们可以更新当前这次遍历的最大长度maxLen = i +1;

(3)若遇到 temp < 0,说明前面的字符串已经不满足条件了,就可以直接return maxLen。

优化:

我们既然要先遍历截取子串,那么' ) '开始的子串一定是不满足条件的,可以直接跳过。

该方法用时是非常高的,在死亡的边缘徘徊!!!

代码:

public class Solution {
    public int longestValidParentheses(String s) {
        int result = 0;
        if(s.length()==0 && s.length()==1) {
            return 0;
        }
        for(int i = 0;i<s.length();i++) {
            if(s.charAt(i) == ')') {
                continue;
            }
            int maxLen = maxValid(s.substring(i));
            if(result < maxLen) {
                result = maxLen;
            }
        }   
        return result;
    }
    
    //计算从字符串串首开始计算最长有效的子串长度
    public int maxValid(String s) {
        int temp = 0;
        int maxLen = 0;
        for(int i = 0;i<s.length();i++) {
            if(s.charAt(i) == '(') {
                temp ++;
                
            }else {
                temp --;
                if(temp<0) {
                    return maxLen;
                }
                if(temp == 0) {
                    maxLen = i+1;
                }
            }
        }
        return maxLen;
    }
}

复杂度分析

  • 时间复杂度: O(n^2)
  • 空间复杂度:O(n)

解法二:动态规划法

这道问题可以使用动态规划法,问题要求的是s[0,length-1]的最长有效子串长度,我们可以用dt[length]数组来统计,dt[]数组的默认值为0,dt[i]表示以i元素结尾的最长子串长度,那么s[i]一定为 ')' 。

在统计过程中,会出现如下几种情况:

(1)当 s[i]== ')'s[i-1] =='('时,例如:"……()",符合条件,我们只需要更新dt[i] = dt[i-2] +2,当然如果 i-2<0的话,直接赋值dt[i]=2

(2)当s[i] == ')'s[i-1] == ')'时,例如:"……))",如果s[i-dt[i-1]-1] = '('

那么
dt[i]=dt[i−1]+dt[i−dt[i−1]−2]+2
解释:

如果最后一个元素为' ) '是有效子串的一部分,那么他一定有一个对应的' ( ' 在倒数第二个 ')' 所在的有效子字符串的前面,如图所示:

代码:

public class Solution {
    public int longestValidParentheses(String s) {
        if(s.length()==0 || s.length()==1) {
            return 0;
        }
        int[] dt = new int[s.length()];
        int result = 0;
        for(int i = 1;i<s.length();i++) {
            if(s.charAt(i)==')') {
                if(s.charAt(i-1)=='(') {
                    dt[i] = (i>=2?dt[i-2]:0) + 2;
                }else if(i>dt[i-1] && s.charAt(i-dt[i-1]-1)=='(' ){
                    dt[i] = dt[i-1] + 2 + ((i-dt[i-1]-2)>=0?dt[i-dt[i-1]-2]:0);
                }
            }
            result = Math.max(result, dt[i]);
        }
        return result;
    }
}    

复杂度分析:

  • 时间复杂度: O(n)
  • 空间复杂度: O(n)

解法三:栈操作

算法思路:

我们利用一个栈,里面先存入一个-1,方便后面计算有效子串长度,我们栈中的栈首元素是为后面计算有效长度做标记

遍历字符串,

  • 若遇到'(',就把下标入栈;
  • 若遇到')',就出栈;
    • 若栈为空,说明该子串无效,因为我们的栈首存放的是标记位,并非'('的下标,那么我们就把当前的i入栈,作为新子串的标记位;
    • 若栈不为空,说明该字串有效,那么更新最长子串长度result = Math.max(result, i-stack.peek());

该算法的巧妙之处,正是栈首存放的是标记位,而非记录'('的下标。

代码:

public class Solution {
    public int longestValidParentheses(String s) {
        Stack<Integer> stack = new Stack<>();
        int result = 0;
        int len = 0;
        stack.add(-1);
        for(int i = 0;i<s.length();i++) {
            if(s.charAt(i)=='(') {
                stack.add(i);
            }else {
                stack.pop();
                if(stack.isEmpty()) {
                    stack.push(i);
                }else {
                    result = Math.max(result, i-stack.peek());
                }
            }
        }
        return result;
    }
}

复杂度分析:

  • 时间复杂度: O(n)
  • 空间复杂度: O(n)

相关文章

  • 32. 最长有效括号

    32. 最长有效括号 视频讲解挺好的

  • 最长有效括号

    32. 最长有效括号 题目: 给你一个只包含 '(' 和 ')' 的字符串,找出最长有效(格式正确且连续)括号子串...

  • 32.最长有效括号 Python 暴力+栈+贪心 三解题!谁说P

    32.最长有效括号[https://leetcode-cn.com/problems/longest-valid-...

  • 32. 最长有效括号

    32. 最长有效括号 题目链接:https://leetcode-cn.com/problems/longest-...

  • leetcode 32 最长有效括号

    32. 最长有效括号 给定一个只包含 '(' 和 ')' 的字符串,找出最长的包含有效括号的子串的长度。 示例 1...

  • LeetCode 32. 最长有效括号 | Python

    32. 最长有效括号 题目来源:力扣(LeetCode)https://leetcode-cn.com/probl...

  • 32. 最长有效括号

    一、题目 给定一个只包含 '(' 和 ')' 的字符串,找出最长的包含有效括号的子串的长度。 示例 1: 输入: ...

  • 32. 最长有效括号

    给定一个只包含 '(' 和 ')' 的字符串,找出最长的包含有效括号的子串的长度。 示例 1: 示例 2: 思路-...

  • 32. 最长有效括号

    自己解法 看到这题的第一想法就是他喵的括号又见括号,之前括号有效性的精髓就是左括号个数要大于右括号个数,所以第一想...

  • 32. 最长有效括号

    题目给定一个只包含 '(' 和 ')' 的字符串,找出最长的包含有效括号的子串的长度。 示例 1:输入: "(()...

网友评论

      本文标题:32. 最长有效括号

      本文链接:https://www.haomeiwen.com/subject/laiqwrtx.html