力扣32(java)-最长有效括号(困难)
题目:
给你一个只包含 '(' 和 ')' 的字符串,找出最长有效(格式正确且连续)括号子串的长度。
示例 1:
输入:s = "(()"
输出:2
解释:最长有效括号子串是 "()"
示例 2:
输入:s = ")()())"
输出:4
解释:最长有效括号子串是 "()()"
示例 3:
输入:s = ""
输出:0
提示:
0 <= s.length <= 3 * 104
s[i] 为 '(' 或 ')'
来源:力扣(LeetCode)
链接:https://leetcode.cn/problems/longest-valid-parentheses
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。
解题思路:
一、栈
参照@笨猪爆破组 大佬的题解
由于该题需要求的是最长有效括号的子串长度,那么栈中不需要存放左括号,而是存放左括号的下标。当遇到右括号时,就弹出栈顶的左括号索引,然后就更新有效长度 = 当前右括号的索引 - 栈顶左括号的索引 + 1,在更新一下最大有效长度

但是当在索引6遇到右括号时,此时栈为空,找不到与之匹配的左括号了,这时前面索引5计算的长度为4,但是实际最大长度为6,这时不知该如何继续计算...
所以就需要在最开始的时候栈里设置一个“参照物”为-1,计算有效长度 = 当前右括号的索引 - 出栈后新的栈顶索引,在后面栈为空时遇到右括号,就将右括号的索引作为新的参照物压入栈中。

前面步骤省略
当遇到索引5时,有效长度为= 5 - (-1) = 6;
当遇到索引6时,就会将栈顶元素 -1 弹出,栈为空,将6压栈;
当遇到索引7时,7入栈;
当遇到索引8时,7出栈,有效长度= 8 - 6 = 2,更新最大长度仍为6;
当遇到索引9时,弹出栈顶元素6,栈为空,将9压栈;
当遇到索引10时,10入栈;
当遇到索引11时,10出栈,有效长度 = 11 - 9 = 2,更新最大长度仍为6;
遍历完毕,返回最大长度6。
代码:
1 class Solution {
2 public int longestValidParentheses(String s) {
3 int n = s.length();
4 Deque<Integer> stack = new ArrayDeque<>();
5 //最先让-1入栈作为参照物
6 stack.addLast(-1);
7 int max = 0;
8 for(int i = 0; i < n; i++){
9 char c = s.charAt(i);
10 if(c == '('){
11 stack.addLast(i);
12 }else{
13 stack.pollLast();
14 if(stack.isEmpty()){
15 //栈为空就把右括号索引入栈,作为新的参照物
16 stack.addLast(i);
17 }else{
18 max = Math.max(max,i - stack.peekLast());
19 }
20 }
21 }
22 return max;
23 }
24 }

二、动态规划(不好理解)
定义dp[i] : 以s[i]结尾的最长有效子串的长度。
子串的末位s[i] 要么是 '(',要么是 ')',下面分情况讨论:
1.s[i]是 '(',以左括号结尾一定不是有效子串,即 dp[i] = 0;
2.s[i]是 ')',这时候就需要考虑前一个子串的末位 s[i-1]:
- 如果 s[i-1] 是 '(',刚好与 s[i] 组成一对,最长有效长度至少也得为2,这时候又需要考虑再前面一个子串末位 s[i-2]:
- 如果 s[i-2]不存在,则有效长度为2,即 dp[i] = 2;
- 如果s[i-2]存在,则需要加上以s[i-2]结尾的有效长度,即dp[i] = dp[i-2] + 2;

- 如果 s[i-1]是 ')',以s[i-1]结尾的最长有效长度为 dp[i-1],这时候就需要看第 i -dp[i-1] - 1这个位置的字符即 s[i-dp[i-1]-1]:
- 如果 s[i-dp[i-1]-1]处的字符不存在或者为 ')',就不能与s[i]处的 ')'配对,即dp[i] = 0;
- 如果 s[i-dp[i-1]-1]处的字符为 '(',与s[i]处的 ')'配对,这时候dp[i] 至少也得为dp[i-1] +2,这时候就看s[i-dp[i-1]-1-1] (s[i-dp[i-1]-2])是否存在:
- s[i-dp[i-1]-2] 存在,dp[i] = dp[i-dp[i-1] -2] + dp[i-1] + 2;
- s[i-dp[i-1]-2]不存在,dp[i] = dp[i-1] + 2;

代码:
1 class Solution {
2 public int longestValidParentheses(String s) {
3 int n = s.length();
4 int ans = 0;
5 int[] dp = new int[n];
6 for(int i = 1; i < n; i++){
7 if(s.charAt(i) == ')'){
8 if(s.charAt(i-1) == '('){
9 dp[i] = (i >= 2) ? dp[i-2]+2 : 2;
10 }else if(i - dp[i-1] > 0 && s.charAt(i- dp[i-1] -1)== '('){
11 //保证第i - dp[i-1]这个位置之前还有字符且前一个字符还必须为左括号
12 dp[i] = dp[i-1] + 2 + (i-dp[i-1] - 2 >= 0 ? dp[i-dp[i-1] -2] : 0);
13 }
14 ans = Math.max(ans, dp[i]);
15 }
16 }
17 return ans;
18 }
19 }

力扣32(java)-最长有效括号(困难)的更多相关文章
- 32、最长有效括号 | 算法(leetode,附思维导图 + 全部解法)300题
零 标题:算法(leetode,附思维导图 + 全部解法)300题之(32)最长有效括号 一 题目描述 二 解法总览(思维导图) 三 全部解法 1 方案1 1)代码: // 方案1 "滑动窗 ...
- 【LeetCode 32】最长有效括号
题目链接 [题解] 设dp[i]表示以第i个字符结尾的最长有效括号的长度. 显然只要考虑s[i]==')'的情况 则如果s[i-1]=='(',则dp[i] = dp[i-2]+2; 如果s[i-1] ...
- Leetcode(32)-最长有效括号
给定一个只包含 '(' 和 ')' 的字符串,找出最长的包含有效括号的子串的长度. 示例 1: 输入: "(()" 输出: 2 解释: 最长有效括号子串为 "()&quo ...
- 力扣(LeetCode)长按键入 个人题解
你的朋友正在使用键盘输入他的名字 name.偶尔,在键入字符 c 时,按键可能会被长按,而字符可能被输入 1 次或多次. 你将会检查键盘输入的字符 typed.如果它对应的可能是你的朋友的名字(其中一 ...
- Java实现 LeetCode 32 最长有效括号
32. 最长有效括号 给定一个只包含 '(' 和 ')' 的字符串,找出最长的包含有效括号的子串的长度. 示例 1: 输入: "(()" 输出: 2 解释: 最长有效括号子串为 & ...
- LeetCode 32. 最长有效括号(Longest Valid Parentheses) 31
32. 最长有效括号 32. Longest Valid Parentheses 题目描述 给定一个只包含 '(' 和 ')' 的字符串,找出最长的包含有效括号的子串的长度. 每日一算法2019/6/ ...
- LeetCode--032--最长有效括号(java)
给定一个只包含 '(' 和 ')' 的字符串,找出最长的包含有效括号的子串的长度. 示例 1: 输入: "(()" 输出: 2 解释: 最长有效括号子串为 "()&quo ...
- 【LeetCode-面试算法经典-Java实现】【032-Longest Valid Parentheses(最长有效括号)】
[032-Longest Valid Parentheses(最长有效括号)] [LeetCode-面试算法经典-Java实现][全部题目文件夹索引] 原题 Given a string contai ...
- C++双指针滑动和利用Vector实现无重复字符的最长子串—力扣算法
题目: 力扣原题链接:https://leetcode-cn.com/problems/longest-substring-without-repeating-characters/ 给定一个字符串, ...
- [LeetCode] 32. Longest Valid Parentheses 最长有效括号
Given a string containing just the characters '(' and ')', find the length of the longest valid (wel ...
随机推荐
- Prompt进阶2:LangGPT(构建高性能Prompt策略和技巧)--最佳实践指南
Prompt进阶2:LangGPT(构建高性能Prompt策略和技巧)--最佳实践指南 0.前言 左图右图 prompt 基本是一样的,差别只在提示工程这个词是否用中英文表达.我们看到,一词之差,回答 ...
- 安装VMware——Unable to install all modules.See log /tmp/vmware-han/vmware-6098.log for detalls.(Exit code 1)的解决方法(模块加载失败)
这是编译失败的原因在VM社区有这样一种解决方案,亲测有效,帮助我自己解决的麻烦,所有在这分享,希望能够帮助到小伙伴:不要被接下来的代码吓倒因为这是github上项目,所以要先在ubuntu上安装git ...
- Java/Kotlin Double保留小数点后几位
下面以保留2位小数,且按照四舍五入规则的例子 方法 1.BigDecimal.setScale() 此方法得到的还是个double数值 double one = 5.864; BigDecimal t ...
- github无法访问?vscode 无法使用github登录同步? 改 hosts 吧
一.无法访问 github.com ? 想要去 github.com 上拿来主义,结果访问不了,或者 npm 时一直失败? 是什么原因不用问,直接使用 https://tool.lu/ip/ 查询到 ...
- day09-数据格式化&验证以及国际化
数据格式化&验证以及国际化 1.数据格式化 1.1基本介绍 说明:在我们提交数据(比如表单时),SpringMVC 是怎样对提交的数据进行转换和处理的 基本数据类型可以和字符串之间自动进行转换 ...
- undefined reference to vtable for "xxx::xxx" in QT(已解决)
PS:要转载请注明出处,本人版权所有. PS: 这个只是基于<我自己>的理解, 如果和你的原则及想法相冲突,请谅解,勿喷. 前置说明 本文发布于 2015-02-09 15:37:25 ...
- 5G+云渲染:如何快速推进XR和元宇宙实现?
XR(扩展现实)领域正在以惊人的速度增长.目前,到 2024 年,一些专家表示这个行业的价值将达到 3000 亿美元. 这个行业发展如此迅速的部分原因是 XR 将在商业环境中的带来巨大利益.近年来,很 ...
- 汽车VR虚拟仿真技术如何加速自动驾驶的发展?
虚拟现实和虚拟仿真将带领自动驾驶汽车从汽车研发.体验.展厅.销售等各个环节迈入全新时代.2019 年,全球增强现实和虚拟现实市场为168 亿美元,到 2023 年,该市场的未来增长预计将超过 1600 ...
- TP6框架--EasyAdmin学习笔记:定义路由
这是我写的学习EasyAdmin的第二章,这一章我给大家分享下如何定义一条路由 正常的tp6定义路由方法如下: /route/admins/app.php 文件内容 //路由变量自定义 Route:: ...
- 记录--图解 Vue 响应式原理
这里给大家分享我在网上总结出来的一些知识,希望对大家有所帮助 最近部门分享,有同学提到了 Vue 响应式原理,大家在讨论时,发现一些同学对这一知识理解还不够深入,不能形成一个闭环,为了帮助大家理解这个 ...