leetcode32. 最长有效括号
给定一个只包含 '('
和 ')'
的字符串,找出最长的包含有效括号的子串的长度。
示例 1:
输入: "(()"
输出: 2
解释: 最长有效括号子串为 "()"
示例 2:
输入: ")()())"
输出: 4
解释: 最长有效括号子串为 "()()"
方法一:动态规划
思路:
本题最暴力的方法就是枚举所有的子串,然后判断这个子串是否是有效的括号,维护最大长度的括号即可。这种想法的时间复杂度为O(N ^ 3),枚举为O(N ^ 2),判断还需要O(N),不能通过。
因为在上面遍历过程中,需要很多的重复计算,所以我们使用动态规划的方法来解决。
我们维护一个dp数组,dp[i]表示**以s[i]为结尾的最长有效括号的长度。**我们可以知道,s中的左括号的位置,对应的dp值为0。
下面我们找状态转移方程,我们遍历s,忽略’(’,因为肯定为0,只考虑’)’:
- 如果s[i-1] = ‘(’,那么s[i-1]和s[i]构成了一个有效括号’()’,再加上dp[i-2]即可,即dp[i] = dp[i-2] + 2。
- 如果s[i-1] = ‘)’,那么此时连续两个右括号,相当于这种情况
'xxxx((。。。。。。))'
,此时遍历到最后的’)’,内层是满足条件的,长度为dp[i-1],需要看这个dp[i-1]所对应的有效括号的前面一个字符,是不是’(’,如果是的话,这样就相当于在dp[i-1]外层加一层’()’,dp[i-1] + 2。同时还需要考虑xxx是不是有效括号,如果是的话也要加上,长度为dp[i-dp[i-1]-2]。 所以dp[i] = dp[i-1] + 2 + dp[i-dp[i-1]-2]
考虑边界条件,只需要把dp[0]置为0即可,因为s的第一个元素对应的肯定是无效的。
我们在填写dp的时候,不断维护保存dp的最大值,返回即可。
- 只遍历了一次,时间复杂度为O(N)。
- 使用了一个dp数组,空间复杂度为O(N)。
代码:
class Solution:
def longestValidParentheses(self, s: str) -> int:
n = len(s)
#初始化dp
dp = [0 for _ in range(n)]
res = 0
#只有一个符号,返回0
if n