[leetcode] 32. 最长有效括号

文章目录

  • 题目描述
  • 解题方法
    • 方法一:栈
      • java代码
      • 复杂度分析
    • 方法二:贪心
      • java代码
      • 复杂度分析
  • 相似题目

题目描述

给你一个只包含 '('')' 的字符串,找出最长有效(格式正确且连续)括号子串的长度。

示例 1:

输入:s = "(()"
输出:2
解释:最长有效括号子串是 "()"

示例 2:

输入:s = ")()())"
输出:4
解释:最长有效括号子串是 "()()"

示例 3:

输入:s = ""
输出:0

提示:

  • 0 <= s.length <= 3 * 104
  • s[i] 为 ‘(’ 或 ‘)’

解题方法

方法一:栈

利用栈先进后出的特性,每次匹配到'('字符,将'('字符的下标位置入栈;当匹配到')' 字符时,将栈顶一个'('字符的下标位置出栈,出栈的'('字符就是对应')'字符匹配的左括号。设匹配到')' 字符下标位置为j,对应出栈'('字符下标位置为ij - i + 1即为当前的’)'字符匹配的括号长度。

通过上面这种方式我们能求出所有’)'对应'('的括号长度,取其中的最大值就是最长有效括号子串的长度吗?答案是否定的。还有一种情况,虽然匹配到了当前')'匹配到了'(',但是与匹配的'('相邻的前面字符串中也可能存在有效的括号子串。此时,就需要将前面子串的括号长度相加。

举个例子,设字符串为" ( ) ( ) \color {red}()\color {green}() ()()",此时绿色的括号匹配长度是2,但我们还需要加上前面红色括号的匹配长度,才能求出最长有效括号子串的长度。

友情提示:以下是正确思路

那我们需要怎么做呢?其实也很简单,只需要提供一个变量start,用来记录有效括号子串的起始位置。初始栈为空时,加入的第一个'('的下标即为有效括号子串的起始位置。而当栈中没有'(',而又匹配到')'时,原来的start位置废弃,后面加入的第一个'('的下标即为新的start。那么当匹配到')'时,如何计算最长有效括号子串的长度呢?分以下两种情况讨论。

  • 第一种,当匹配到的字符')'有对应出栈'('字符,且'('出栈后栈不为空。设当前')'下标为j'('出栈后栈顶元素为i,则当前')'匹配的有效括号子串的长度为j - i
  • 第二种,当匹配到的字符')'有对应出栈'('字符,且'('出栈后栈为空。设当前')'下标为j,则当前')'匹配的有效括号子串的长度为j - start + 1

取上面两种情况最大值即为最长有效括号子串的长度。

java代码

public int longestValidParentheses(String s) {
    if (s == null || s.length() == 0) {
        return 0;
    }
    // 记录有效子串的起始位置
    int start = 0;
    int max = 0;
    Stack<Integer> stack = new Stack<Integer>();
    for (int i = 0; i < s.length(); i++) {
        char c = s.charAt(i);
        if (c == '(') {
            stack.push(i);
        } else if (c == ')') {
            if (stack.isEmpty()) {
                start = i + 1;
            } else {
                stack.pop();
                if (stack.isEmpty()) {
                    // 栈为空时,有效子串的长度 = 右括号的位置 - 有效子串的起始位置 + 1
                    max = Math.max(max, i - start + 1);
                } else {
                    // 栈不为空时,有效子串的长度 = 右括号的位置 - 栈顶元素的位置,因为栈顶元素的下一个位置是匹配的'('
                    max = Math.max(max, i - stack.peek());
                }
            }
        }

    }
    return max;
}

复杂度分析

时间复杂度: O ( N ) O(N) O(N) N N N为字符串长度。需要遍历一次字符串。
空间复杂度: O ( N ) O(N) O(N),需要提供栈的存储空间。

方法二:贪心

方法一需要用栈存储左括号位置,那么有没有一种方法只使用 O ( 1 ) O(1) O(1)的常数空间呢?答案是有的,不过需要遍历两次字符串,我说一下思路。

第一次遍历,我们需要从头到尾遍历字符串,我们记 l e f t left left为匹配的'('数量, r i g h t right right为匹配的')'数量数量。当 r i g h t > l e f t right>left right>left时, l e f t left left r i g h t right right置为 0 0 0;当 r i g h t = = l e f t right==left right==left时,计算当前字符串的有效长度为 2 × r i g h t 2 \times right 2×right,并且记录目前为止最长字符串长度。此次遍历,有一种情况没有考虑到。例如字符串" ( ( ) (() (()",当 l e f t left left一直大于 r i g h t right right时,是求不出最长长度的。

所以需要第二次遍历,这次我们从尾到头遍历字符串。还是记 l e f t left left为匹配的'('数量, r i g h t right right为匹配的')'数量数量。当 l e f t > r i g h t left>right left>right时, l e f t left left r i g h t right right置为 0 0 0;当 l e f t = = r i g h t left==right left==right时,计算当前字符串的有效长度为 2 × l e f t 2 \times left 2×left,并且记录目前为止最长字符串长度。

java代码

public int longestValidParentheses(String s) {
    int left = 0, right = 0;
    int max = 0;
    for (int i = 0; i < s.length(); i++) {
        char c = s.charAt(i);
        if (c == '(') {
            left += 1;
        } else {
            right += 1;
        }
        if (right > left) {
            left = 0;
            right = 0;
        } else if (right == left) {
            max = Math.max(right * 2, max);
        }
    }
    left = right = 0;
    for (int i = s.length() - 1; i >= 0; i--) {
        char c = s.charAt(i);
        if (c == '(') {
            left += 1;
        } else {
            right += 1;
        }
        if (left > right) {
            left = 0;
            right = 0;
        } else if (left == right) {
            max = Math.max(left * 2, max);
        }
    }
    return max;
}

复杂度分析

时间复杂度: O ( N ) O(N) O(N) N N N为字符串长度。需要遍历两次字符串。
空间复杂度: O ( 1 ) O(1) O(1),只需要存储常数级别的变量。

相似题目

[leetcode] 20. 有效的括号


  • 个人公众号
    个人公众号
  • 个人小游戏
    个人小游戏

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/379965.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

[Java][算法 哈希]Day 01---LeetCode 热题 100---01~03

LeetCode 热题 100---01~03 ------->哈希 第一题 两数之和 思路 最直接的理解就是 找出两个数的和等于目标数 这两个数可以相同 但是不能是同一个数字&#xff08;从数组上理解就是内存上不是同一位置&#xff09; 解法一&#xff1a;暴力法 暴力解万物 按照需求 …

spring-security authentication persistence

翻译版本【spring-security 6.2.1】persistence Persisting Authentication 用户第一次请求受保护的资源时&#xff0c;系统会提示他们输入凭据。提示输入凭据的最常见方法之一是将用户重定向到登录页面。未经身份验证的用户请求受保护的资源的HTTP交换可能如下所示: 例1。未…

前端实现支付跳转以及回跳

// 支付地址 const baseURL http://pcapi-xiaotuxian-front-devtest.itheima.net/ const backURL http://127.0.0.1:5173/paycallback const redirectUrl encodeURIComponent(backURL) const payUrl ${baseURL}pay/aliPay?orderId${route.query.id}&redirect${redirec…

PyTorch 2.2 中文官方教程(一)

PyTorch 秘籍 PyTorch 秘籍 原文&#xff1a;pytorch.org/tutorials/recipes/recipes_index.html 译者&#xff1a;飞龙 协议&#xff1a;CC BY-NC-SA 4.0 秘籍是关于如何使用特定 PyTorch 功能的简短、可操作的示例&#xff0c;与我们的全长教程不同。 PyTorch 原型示例 原文…

Linux嵌入式开发+驱动开发-中断

swi汇编指令可以产生软中断&#xff0c;以下是硬件中断的产生到执行完毕的全过程&#xff1a; 在自己设计的芯片“CPU响应中断”程序的第四个步骤可以转向“中断向量控制器”&#xff0c;中断向量控制器中存储中断元服务地址即处理中断处理程序的地址&#xff0c;而不用使用0X1…

ONLYOFFICE文档8.0新功能浅探

ONLYOFFICE文档8.0新功能浅探 上个月末这个月初的几天&#xff0c;ONLYOFFICE版本更新了&#xff01;更新到了一个比较整的大的版本号&#xff0c;8.0版本&#xff0c;看来这个生产力工具的升级速度基本上能保持每年两个版本号的速度&#xff0c;还是很快的&#xff0c;一般来…

Java强训day15(选择题编程题)

选择题 自连接使用一张表编程题 题目1 import java.util.Scanner;public class Main { public static int res(int n) {StringBuffer s new StringBuffer();while(n!0) {s.append(n%2);n/2;}int sum 0;String ss s.reverse().toString();for(int i0;i<ss.length()…

秋招上岸大厂,分享一下经验

文章目录 秋招过程学习过程项目经验简历经验面试经验offer选择总结 秋招过程 今天是除夕&#xff0c;秋招已经正式结束了&#xff0c;等春节过完就到了春招的时间点了。 运气比较好&#xff0c;能在秋招的末尾进入一家大厂&#xff0c;拿到20k的sp offer。 从九月份十月份就开…

TCP 传输控制协议——详细

目录 1 TCP 1.1 TCP 最主要的特点 1.2 TCP 的连接 TCP 连接&#xff0c;IP 地址&#xff0c;套接字 1.3 可靠传输的工作原理 1.3.1 停止等待协议 &#xff08;1&#xff09;无差错情况 &#xff08;2&#xff09;出现差错 &#xff08;3&#xff09;确认丢失和确认迟到…

【RT-DETR改进涨点】更加聚焦的边界框损失Focaler-IoU、InnerFocalerIoU(二次创新)

一、本文介绍 本文给大家带来的改进机制是更加聚焦的边界框损失Focaler-IoU已经我进行二次创新的InnerFocalerIoU同时本文的内容支持现阶段的百分之九十以上的IoU,比如Focaler-IoU、Focaler-ShapeIoU、Inner-Focaler-ShapeIoU包含非常全的损失函数,边界框的损失函数只看这一…

RCE(命令执行)知识点总结最详细

description: 这里是CTF做题时常见的会遇见的RCE的漏洞知识点总结。 如果你觉得写得好并且想看更多web知识的话可以去gitbook.22kaka.fun去看&#xff0c;上面是我写的一本关于web学习的一个gitbook&#xff0c;当然如果你能去我的github为我的这个项目点亮星星我会感激不尽htt…

C#用Array类的Reverse方法反转数组中元素

目录 一、Array.Reverse 方法 1.重载 2.Reverse(Array, Int32, Int32) 3. Reverse(Array) 4.Reverse(T[]) 5. Reverse(T[], Int32, Int32) 二、实例 1.Array.Reverse 方法4种重载方法综合实例 2.Reverse(Array)方法的实例 一、Array.Reverse 方法 反转一维 Array 或部…

Android修改系统默认字体

文章目录 前言一、方案1、将定制的custom_fonts.xml配置文件编译到系统中2、将自定义的字体ttf文件编译到系统中3、在系统的编译mk中添加fonts.mk的引用4、修改系统代码,使得优先加载使用custom_fonts.xml前言 Android系统中的字体配置文件为/system/etc/fonts.xml 关于fonts…

JVM之GC垃圾回收

GC垃圾回收 如何判断对象可以回收 引用计数法 如果有对象引用计数加一&#xff0c;没有对象引用&#xff0c;计数减一&#xff0c;如果计数为零&#xff0c;则回收 但是如果存在循环引用&#xff0c;即A对象引用B对象&#xff0c;B对象引用A对象&#xff0c;会造成内存泄漏 可…

linux之wsl2安装远程桌面

0. 安装后的效果 1. wsl中打开terminal并安装库 sudo apt-get purge xrdp sudo apt install -y xrdp sudo apt install -y xfce4 sudo apt install -y xfce4-goodies 2.优化显示 sudo sed -i s/max_bpp32/#max_bpp32\nmax_bpp128/g /etc/xrdp/xrdp.ini sudo sed -i s/xserverbp…

听说有 Hugging Face 陪伴的春节,是这样的…

辞旧迎新春节到&#xff0c;家家户户好热闹。Hugging Face 中国团队成员祝各位社区成员们新春快乐&#xff0c;万事如意&#xff01; 过去的一年我们持续看到 AI 技术的腾飞和发展&#xff0c;以及诸多机构为开源 AI 作出巨大的贡献。非常感谢将模型、数据集和应用 Demo 发布在…

寒假作业:2024/2/8

作业1&#xff1a;现有文件test.c\test1.c\main.c,编写Makkefile Makefile代码&#xff1a; CCgcc EXEa.out OBJS$(patsubst %.c,%.o,$(wildcard *.c)) CFLAGS-c -oall:$(EXE)$(EXE):$(OBJS)$(CC) $^ -o $%.o:%.c$(CC) $(CFLAGS) $ $^.PHONY:cleanclean:rm $(OBJS) $(EXE)效果…

网络安全产品之认识准入控制系统

文章目录 一、什么是准入控制系统二、准入控制系统的主要功能1. 接入设备的身份认证2. 接入设备的安全性检查 三、准入控制系统的工作原理四、准入控制系统的特点五、准入控制系统的部署方式1. 网关模式2. 控制旁路模式 六、准入控制系统的应用场景七、企业如何利用准入控制系统…

CTF--Web安全--SQL注入之Post-Union注入

一、手动POST注入实现绕过 账号密码检测 我们利用sqli-labs/Less-11靶场来进行演示&#xff1a; 我们可以看到一个登录页面 打开Less-11的根目录&#xff0c;我们打开页面的源代码(PHP实现)。 用VS-code打开文件&#xff0c;找到验证登录信息的代码行。 此形式的代码存在POST…

【Linux】Linux开发工具(yum、gdb、git)详解

一、软件包管理器 yum 1、什么是软件包 在 Linux 下安装软件&#xff0c;通常的办法是下载到程序的源代码&#xff0c;并进行编译&#xff0c;得到可执行程序。但这样太麻烦了&#xff0c;于是有些人把一些常用的软件提前编译好&#xff0c;做成软件包&#xff08;可以理解成…