方法一:栈

遍历字符串,使用栈来存储左括号对应的下标

1、对于遇到的每个‘(’ ,将它的下标放入栈中

2、对于遇到的每个 ‘)’ ,先弹出栈顶元素表示匹配了当前右括号:

(1)、如果栈为空,说明当前的右括号为没有被匹配的右括号,将其下标放入栈中来更新我们之前提到的「最后一个没有被匹配的右括号的下标」

(2)、如果栈不为空,当前右括号的下标减去栈顶元素即为「以该右括号为结尾的最长有效括号的长度」

3、从前往后遍历字符串并更新答案即可。

需要注意的是,如果一开始栈为空,第一个字符为左括号的时候我们会将其放入栈中,这样就不满足提及的「最后一个没有被匹配的右括号的下标」,为了保持统一,我们在一开始的时候往栈中放入一个值为 -1 的元素。

时间复杂度:o(n)

空间复杂度:o(n)

class Solution {
  public:
    int longestValidParentheses(string s) {
        stack<int> res;
        // 子串长度,初始化为0
        int len = 0;
        // 记录上一次有效子串结束的位置,为了保持统一初始化为-1
        int start = -1;

        for (int i = 0; i < s.length(); i++) {
            if (s[i] == '(') {
                res.push(i);
            } else {
                if (res.empty()) {
                    start = i;
                } else {
                    res.pop();
                    if (res.empty()) {
                        len = max(len, i - start);
                    } else {
                        len = max(len, i - res.top());
                    }
                }
            }
        }
        return len;
    }
};

方法二:动态规划

创建大小为s.length()的数组dp,记录以s[i]为结尾的有效子串的长度。

有效子串一定是以右括号为结尾,所以遍历字符串,只有当前字符为右括号时才进行长度判断。

有两种情况:

1、s[i] = ')' 且 s[i-1] = '(',表示字符串形如 ‘......()’,则可推出:

dp[i] = dp[i-2] + 2

以进行这样的转移,是因为结束部分的 "()" 是一个有效子字符串,并且将之前有效子字符串的长度增加了 2

2、s[i] = ')' 且 s[i-1] = ')',表示字符串形如 ‘......))’,则可推出:

如果s[i - dp[i-1] - 1] = '(',则

dp[i] = dp[i-1] + dp[i-dp[i-1]-2] + 2

考虑如果倒数第二个 ‘)’ 是一个有效子字符串的一部分(记作 subs),对于最后一个‘)’ ,如果它是一个更长子字符串的一部分,那么它一定有一个对应的 ‘(’ ,且它的位置在倒数第二个 ‘)’ 所在的有效子字符串的前面(也就是 subs的前面)。因此,如果子字符串 subs 的前面恰好是 ‘(’ ,那么我们就用 2 加上 subs 的长度(dp[i−1])去更新 dp[i]。同时,也会把有效子串 “(subs)” 之前的有效子串的长度也加上,也就是再加上 dp[i−dp[i−1]−2]。

时间复杂度:o(n)

空间复杂度:o(n)

class Solution {
  public:
    int longestValidParentheses(string s) {
        vector<int> dp(s.length(), 0);
        int len = 0;
        for (int i = 1; i < s.length(); i++) {
            // 有效的子串一定是以右括号结尾
            if (s[i] == ')') {
                if (s[i - 1] == '(') {
                    dp[i] = (i >= 2 ? dp[i - 2] : 0) + 2;
                } else {
                    if (s[i - dp[i - 1] - 1] == '(' && i - dp[i - 1] > 0) {
                        dp[i] = dp[i - 1] +
                                ((i - dp[i - 1] >= 2) ? dp[i - dp[i - 1] - 2] : 0) + 2;
                    }
                }
            }
            len = max(len, dp[i]);
        }
        return len;
    }
};