LeetCode.32最长有效括号详解

news2025/1/12 20:48:28

问题描述

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

解题思路1

有效的括号字符串意味着每一个左括号 '(' 都可以找到一个相匹配的右括号 ')'。栈可以帮助我们追踪尚未匹配的括号,并有效地处理嵌套结构。

解题步骤如下:

  1. 初始化一个栈,并将一个特殊的索引 -1 压入栈。这一步是为了方便计算最长有效子串的长度。
  2. 遍历字符串的每个字符:
    • 如果字符是 '(',将其索引压入栈。
    • 如果字符是 ')':
      • 弹出栈顶元素(这代表最近的一个 '(' 已经被匹配)。
      • 如果栈为空,说明没有 '(' 可以与当前的 ')' 匹配,将当前的索引压入栈作为新的起点。
      • 如果栈不为空,当前索引减去栈顶元素索引即为有效子串的长度,更新最长有效长度。
  3. 返回最长有效长度

代码实现:

class Solution {
public:
    int longestValidParentheses(string s) {
        int maxLength = 0;
        stack<int> indexStack;
        indexStack.push(-1); // 前一个没有匹配的 ')' 的位置

        for (int i = 0; i < s.length(); i++) {
            if (s[i] == '(') {
                // 将 '(' 的索引入栈
                indexStack.push(i);
            } else {
                // 弹出栈顶元素
                indexStack.pop();
                if (indexStack.empty()) {
                    // 如果栈为空,意味着没有 '(' 可以与当前 ')' 匹配
                    indexStack.push(i); // 当前位置变为最后一个没有匹配的 ')'
                } else {
                    // 栈不为空,则计算有效长度
                    int validLength = i - indexStack.top();
                    maxLength = max(maxLength, validLength);
                }
            }
        }

        return maxLength;
    }
};

这个解题思路主要参考了逆波兰表示法这个知识点。

解题思路2

第二种方法效率更高,但思路没有上面第一种思路清晰。

使用两个计数器

我们采用两个计数器(leftright)来分别跟踪左括号和右括号的数量。这种方法的关键在于通过两次遍历字符串来确保所有的括号都能找到对应的匹配,从而计算出最长的有效括号子串。

第一次遍历:从左到右

在第一次遍历中,我们从字符串的开始到结束进行扫描:

  • 遇到 '(' 时,增加 left 计数器。
  • 遇到 ')' 时,增加 right 计数器。
  • 每当 leftright 数量相等时,我们更新最长有效子串的长度,这是因为到目前为止,所有遇到的括号都能完全匹配。
  • 如果 right 的数量超过 left,这表明括号已经无法匹配,需要重置两个计数器。这是因为任何以 ')' 开始的子串都不可能是有效的。
第二次遍历:从右到左

第二次遍历与第一次遍历类似,但方向相反。这主要是为了捕捉那些在第一次遍历中由于左括号过多而未能正确处理的情况:

  • 遇到 ')' 时,增加 right 计数器。
  • 遇到 '(' 时,增加 left 计数器。
  • leftright 数量相等时,同样更新最长有效子串的长度。
  • 如果 left 数量超过 right,则重置两个计数器,因为任何以 '(' 结尾的子串都不能是有效的。

代码实现:

class Solution {
public:
    int longestValidParentheses(string s) {
    int left = 0, right = 0, maxLength = 0;

    // 第一次遍历:从左到右
    for (int i = 0; i < s.length(); ++i) {
        if (s[i] == '(') {
            left++;
        } else {
            right++;
        }

        if (left == right) {
            maxLength = max(maxLength, 2 * right);
        } else if (right > left) {
            left = 0;
            right = 0;
        }
    }

    left = 0;
    right = 0;

    // 第二次遍历:从右到左
    for (int i = s.length() - 1; i >= 0; --i) {
        if (s[i] == ')') {
            right++;
        } else {
            left++;
        }

        if (left == right) {
            maxLength = max(maxLength, 2 * left);
        } else if (left > right) {
            left = 0;
            right = 0;
        }
    }

    return maxLength;
    }
};

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

缓存雪崩、穿透、击穿的概念和如何避免

雪崩 概念 高并发请求多个key&#xff0c;此时多个key同时失效、不在缓存中&#xff0c;请求全部打到数据库&#xff0c;使数据库无法处理这么多的连接&#xff0c;导致数据库死机 如何避免 防止多个key同时过期&#xff0c;设置随机的过期时间不设置过期时间&#xff0c;有…

编程精粹—— Microsoft 编写优质无错 C 程序秘诀 08:剩下的就是态度问题

这是一本老书&#xff0c;作者 Steve Maguire 在微软工作期间写了这本书&#xff0c;英文版于 1993 年发布。2013 年推出了 20 周年纪念第二版。我们看到的标题是中译版名字&#xff0c;英文版的名字是《Writing Clean Code ─── Microsoft’s Techniques for Developing》&a…

chatglm系列知识

一、目录 chatglm 是什么语言模型与transformer decoder 的区别解释prefix LM与Cause LMchatglm&#xff08;prefix LM&#xff09;与decoder-only LM 核心区别glm 架构chatglm 预训练方式chatglm 微调chatglm与chatglm2、chatglm3的区别chatglm 激活函数采用gelu, 为什么chat…

融资融券账户与普通账户有何区别?一文读懂为什么要开通两融账户

01 融资融券账户与普通账户的区别 1、开通条件不同&#xff1a; ①普通账户&#xff1a;开户的门槛低&#xff0c;一般年满18岁以上就能开通。&#xff08;70岁以上需要临柜开户&#xff09;。 ②融资融券&#xff1a;融资融券的准入门槛相对较高&#xff0c;需要满足以下几…

SQLite3的使用

14_SQLite3 SQLite3是一个嵌入式数据库系统&#xff0c;它的数据库就是一个文件。SQLite3不需要一个单独的服务器进程或操作系统&#xff0c;不需要配置&#xff0c;这意味着不需要安装或管理&#xff0c;所有的维护都来自于SQLite3软件本身。 安装步骤 在Linux上安装SQLite…

python桌面应用

py文件 import osimport wx import wx.html2class MyFrame(wx.Frame):def __init__(self, parent):wx.Frame.__init__(self, parent, title"启动啦", size(1000, 700))# 创建一个Web视图组件self.browser wx.html2.WebView.New(self)# 加载本地HTML文件# self.brow…

WebFlux 和 Spring Security 会碰出哪些火花?

项目创建成功后&#xff0c;我们添加一个接口&#xff0c;用来获取登录用户信息&#xff0c;如下&#xff1a; RestController public class UserController { GetMapping(“/user”) public Mono getCurrentUser(Mono principal) { return principal; } } 注意我们的返…

【Leetcode每日一题】 01背包 - DP41 【模板】01背包(难度⭐⭐)(80)

1. 题目解析 题目链接&#xff1a;DP41 【模板】01背包 这个问题的理解其实相当简单&#xff0c;只需看一下示例&#xff0c;基本就能明白其含义了。 2.算法原理 第一问&#xff1a;不超过总体积的背包问题 1. 状态表示 dp[i][j] 表示&#xff1a;从前 i 个物品中挑选&…

android adb常用命令集

1、系统调试 #adb shell&#xff1a;进入设备的 shell 命令行界面&#xff0c;可以在此执行各种 Linux 命令和特定的 Android 命令。 #adb shell dumpsys&#xff1a;提供关于系统服务和其状态的详细信息。 #adb logcat&#xff1a;实时查看设备的日志信息。可以使用过滤条件来…

Arduino称重传感器和 HX711 放大器(数字秤)

Arduino称重传感器和 HX711 放大器&#xff08;数字秤&#xff09; Arduino with Load Cell and HX711 Amplifier (Digital Scale) In this guide, you’ll learn how to create a digital scale with the Arduino using a load cell and the HX711 amplifier. First, you’l…

二叉树-左叶子之和(easy)

目录 一、问题描述 二、解题思路 三、代码实现 四、刷题链接 一、问题描述 二、解题思路 此题属于树遍历的简单题&#xff0c;用递归深度遍历的方式&#xff0c;当遇到左叶子结点(在递归函数中加上一个判断当前结点是左结点还是右结点的标记位)&#xff0c;此时加上当前结点…

ONLYOFFICE 桌面编辑器 8.1:全新升级,助您轻松高效处理办公文档

ONLYOFFICE 桌面编辑器 一、前言二、轻松编辑器 PDF 文件三、用幻灯片版式快速修改幻灯片四、无缝切换文档编辑、审阅和查看模式五、改进从右至左语言的支持 & 新的本地化选项六、版本 8.1&#xff1a;其他新功能七、ONLYOFFICE 官网&#xff1a;https://www.onlyoffice.co…

OnlyOffice8.1新功能测评

一、导语 时隔四个月&#xff0c;OnlyOffice推出了8.1版本。 四个月过去&#xff0c;笔者的项目也接近尾声&#xff0c;在项目过程中还把OnlyOffice插件推荐给了项目组&#xff0c;希望官方多出好用功能&#xff0c;造福我们广大项目O(∩_∩)O 回归正题&#xff0c;与前几个…

【数据结构与算法】之(数据结构绪论篇)(一)溢彩色

总而言之&#xff1a;《数据结构》是介于数学、计算机硬件和计算机软件三者之间的一门核心课程 1-1.抽象数据类型&#xff1a; 一个数学模型及定义在该模型上的一组操作&#xff1b;抽象数据类型体现了程序设计中问题分解&#xff0c;抽象和信息隐藏的特性。 抽象&#xff1a…

最新《pvz植物大战僵尸杂交版》整合安装包,全面支持Android、ios、Windows,附教程!

今天&#xff0c;阿星要聊聊最近全网大火的一款老游戏——《植物大战僵尸》杂交版。 虽然它不是什么3A大作&#xff0c;但在阿星的心里&#xff0c;它永远是那个让人回味无穷的经典。记得十年前&#xff0c;阿星和大多数玩家一样&#xff0c;玩的都是盗版。那时候的《植物大战…

【人机交互 复习】第1章 人机交互概述

人机交互的知识点碎&#xff0c;而且都是文字&#xff0c;过一遍脑子里什么都留不下&#xff0c;但是背时间已经来不及了&#xff0c;最好还是找题要题感吧&#xff0c;加深印象才是做对文科的关键 一、概念 1.人机交互&#xff08;Human-Computer Interaction,HCI)&#xff1…

路由器ARP和ARP-proxy(华为)

#交换设备 路由器ARP和ARP-proxy(华为) 当一个广播域中的主机想要访问另外一个广播域的主机时&#xff0c;会广播ARP报文&#xff0c;询问目标IP地址所对应的MAC地址&#xff0c;默认情况下&#xff0c;arp记录是设备自动生成的&#xff0c;但是这样会容易受到ARP欺骗攻击&am…

系统架构设计师 - 数据库系统(2)

数据库系统 数据库系统规范化理论 ★ ★ ★ ★ ★函数依赖求候选键Armstrong公理范式判断第一范式 1NF第二范式 2NF第三范式 3NFBC 范式 BCNF 模式分解保持函数依赖分解无损分解 并发控制 ★事务的 ACID 特性并发存在的问题并发解决方案 - 封锁协议 数据库的安全性 ★安全性的分…

C++多线程异步日志实现

使用C11标准&#xff0c;构建了一个方便使用的、轻量化的日志系统。封装线程安全的lockQueue&#xff0c;实现对每条日志添加信息、push到lockQueue中的LogTmp类&#xff0c;实现一个多线程异步的日志系统Logger。 lockqueue.h #pragma once #include <queue> #include…