LeetCode/solutions/32. Longest Valid Parentheses.md
2020-03-31 22:09:20 +08:00

3.5 KiB
Raw Blame History

32. Longest Valid Parentheses

思路

求有效括号匹配子串的最大长度。

思路一、栈

括号匹配类的问题往往都能用栈解决,此题也不例外。

从前往后遍历字符串设下标为i

  • 如果遇到左括号或者栈空则将当前下标i压入栈
  • 如果遇到右括号且栈顶为左括号,则遇到了匹配的括号对,将栈顶弹出即可;

然后就可以计算以字符s[i]作为结尾的最大长度:i - stk.top()

时空复杂度均为O(n)

思路二、动归

由于此问题存在很多子问题,且子问题之前又有很多重复,所可以考虑用动归。

先来看状态定义:

dp[i]: 以字符s[i]结尾的最大匹配长度

状态转移方程就很简单了:

  • 如果s[i] = '(',那么dp[i] = 0
  • 否则,我们需要看以字符s[i-1]为结尾的最大匹配子串的前面那个字符s[i - dp[i-1] - 1]是否是左括号:
    • 若是,说明匹配上了,所以dp[i] = dp[i-1] + 2 + dp[i - dp[i-1] - 2]
    • 若不是,则没匹配上,dp[i] = 0

时空复杂度均为O(n)

思路三

前面两种思路的空间复杂度均为O(n),此题还有一种常数空间的思路,和前面栈的思想有点类似。

使用两个变量left和right分别用来记录到当前位置时左括号和右括号的出现次数当遇到左括号时left自增1否则right自增1。

  • left == right时,说明匹配成功,此时匹配长度为2*left
  • 一旦遇到right > left说明当前位置不能匹配则重新开始即left和right均重置为0。

但是对于"(()"这种时在遍历结束时左右子括号数都不相等此时没法更新结果res但其实正确答案是2怎么处理这种情况呢答案是再反向遍历一遍采取类似的方法稍有不同的是此时若right < left则重置0。

时间复杂度O(n)空间复杂度O(1)

C++

思路一

class Solution {
public:
    int longestValidParentheses(string s) {
        stack<int>stk;
        int res = 0, start = 0;
        for(int i = 0; i < s.size(); i++){
            if(!stk.empty() && s[stk.top()] == '(' && s[i] == ')') stk.pop();
            else stk.push(i);
            // 栈顶为当前匹配子串的起始位置
            res = max(res, stk.empty() ? i + 1 : i - stk.top());
        }
        return res;
    }
};

思路二

class Solution {
public:
    int longestValidParentheses(string s) {
        vector<int>dp(s.size(), 0);
        int res = 0;
        for(int i = 1; i < s.size(); i++){
            if(s[i] == '(') continue;
            int tmp = i - dp[i-1] - 1;
            if(tmp >= 0 && s[tmp] == '(') 
                dp[i] = dp[i-1] + 2 + (tmp >= 1 ? dp[tmp - 1] : 0);
            res = max(res, dp[i]);
        }
        return res;
    }
};

思路三

class Solution {
public:
    int longestValidParentheses(string s) {
        int res = 0, left = 0, right = 0, n = s.size();
        for (int i = 0; i < n; i++) {
            if(s[i] == '(') left++; 
            else right++;
            if (left == right) res = max(res, 2 * right);
            else if (right > left) left = right = 0;
        }
        
        left = right = 0;
        for (int i = n - 1; i >= 0; i--) {
            if(s[i] == '(') left++;
            else right++;
            if (left == right) res = max(res, 2 * left);
            else if (left > right) left = right = 0;
        }
        return res;
    }
};