32. 最长有效括号
3 min
题目
解法:动态规划
思路
动态规划,定义 dp[i] 表示以位置 i 结尾的最长有效括号子串的长度。
当 s[i] == ’)’ 时,有两种情况:若 s[i-1]==’(‘,则形成 ”()“,dp[i] = dp[i-2] + 2;若 s[i-1]==’)‘,则检查 s[i-dp[i-1]-1] 是否为 ’(‘,若是则可扩展前面的有效串。这种方法能正确计算所有可能的有效括号子串长度,因为每个位置的状态都依赖于前面已计算的状态,通过状态转移覆盖了所有可能的匹配情况。
- Step 1: 初始化 dp 数组全为 0,ans 记录最大值
- Step 2: 遍历字符串从索引 1 到 n-1
- Step 3: 如果当前字符是 ’)’
- Step 4: 如果前一个字符是 ’(‘,则 dp[i] = (i >= 2 ? dp[i-2] : 0) + 2
- Step 5: 如果前一个字符是 ’)‘,计算 j = i - dp[i-1] - 1,如果 j >= 0 且 s[j] == ’(‘,则 dp[i] = dp[i-1] + 2 + (j >= 1 ? dp[j-1] : 0)
- Step 6: 更新 ans = max(ans, dp[i])
- Step 7: 返回 ans
复杂度
- 时间复杂度: ,其中 n 是字符串长度,只需要一次遍历
- 空间复杂度: ,用于存储 dp 数组
代码
class Solution {
public:
int longestValidParentheses(string s) {
int n = s.size();
vector<int> dp(n, 0);
int ans = 0;
for (int i = 1; i < n; i++) {
if (s[i] == ')') {
if (s[i - 1] == '(') {
dp[i] = (i >= 2 ? dp[i - 2] : 0) + 2;
} else {
int j = i - dp[i - 1] - 1;
if (j >= 0 && s[j] == '(') {
dp[i] = dp[i - 1] + 2 + (j >= 1 ? dp[j - 1] : 0);
}
}
}
ans = max(ans, dp[i]);
}
return ans;
}
};