作者 | 王磊

来源 | Java中文社群(ID:javacn666)

转载请联系授权(微信ID:GG_Stone)

今天要讲的这道题是 bilibili 今年的笔试真题,也是一道关于栈的经典面试题。

经过前面文章的学习,我想很多朋友已经看出来了,我接下来要写的是一个关于「算法图解」的系列文章,中间可能会穿插少量的其他类型的文章,但「算法和数据结构」会是我今年文章输出的重点内容。

我在写这个算法系列的时候会注意两个问题:

  • 保证算法的解题思路大家都能看懂,因此我会以图片的形式进行思路讲解,这样更直观、更易于理解

  • 在介绍完一个知识点之后,会进行大量的练习,以巩固所学的内容,比如当我讲完「栈」结构之后,我会围绕着「栈」做一系列的经典面试题练习。

学习算法最关键的是掌握解题的思路,只要思路对了,编写代码只是时间的问题。

我们先来回顾一下往期关于「栈」的内容:

  • 《动图演示:手撸堆栈的两种实现方法!》

  • 《JDK 竟然是这样实现栈的?》

  • 《链表反转的两种实现方法,后一种击败了100%的用户!》

  • 《算法图解:如何找出栈中的最小值?》

那么接下来,我们就进入今天的正式内容...

题目

给定一个只包括 '('')''{''}''['']' 的字符串,判断字符串是否有效。

有效字符串需满足:

  • 左括号必须用相同类型的右括号闭合。

  • 左括号必须以正确的顺序闭合。

  • 注意空字符串可被认为是有效字符串。

示例 1:

输入: "()"

输出: true

示例 2:

输入: "()[]{}"

输出: true

示例 3:

输入: "(]"

输出: false

示例 4:

输入: "([)]" 输出: false

示例 5:

输入: "{[]}"

输出: true

LeetCode 地址:https://leetcode-cn.com/problems/valid-parentheses

解题思路

这道题考察的是就是验证括号的对称性,比如“([{}])”这种字符串就是正确的,应该返回 true,而“([{})]”这种字符串就是错误的,应该返回 false。

从上面的题目可以看出,括号总共分为三类:小括号、中括号和大括号,那么我们可以利用栈先进后出的特性,将所有左边的括号(“(”、“[”、“{”)先入栈,然后再碰到右括号时,让它与栈顶的元素进行匹配,比如当遇到“)”时,如果栈顶是“(”,则说明匹配成功,栈顶元素出栈再继续字符串循环的流程,如果匹配错误就直接返回 false。

假设我们要匹配字符串“(([]))”是否合法?那么执行流程就是这样的。

首先遇到左边括号,先入栈:

接下来又是左边括号,继续入栈:

然后又是左边括号,继续入栈:

接下来是右边括号,与栈顶元素匹配,“[]”为一对合法的括号,匹配成功栈顶元素出栈:

接下来又是右边括号,与栈顶元素匹配,“()”为一对合法的括号,匹配成功栈顶元素出栈:

接下来又是右边括号,与栈顶元素匹配,“()”为一对合法的括号,匹配成功栈顶元素出栈:

当字符串循环结束并且栈为空栈时,则证明此字符串的括号匹配合法,最终的效果如下图所示:

那么接下来我们就用代码来实现一下整个过程...

实现代码一

public boolean isValid(String s) {int slen = s.length(); // 括号的长度if (slen % 2 == 1) { // 括号不是成对出现直接返回 falsereturn false;}// 把所有对比的括号存入 map,对比时用Map<Character, Character> map = new HashMap<>();map.put(')', '(');map.put('}', '{');map.put(']', '[');// 定义栈,用于存取括号(辅助比较)Stack<Character> stack = new Stack<>();for (int i = 0; i < slen; i++) { // 循环所有字符char c = s.charAt(i);if (map.containsKey(c)) { // 为右边的括号,如 ')'、'}' 等if (stack.isEmpty() || stack.peek() != map.get(c)) { // 栈为空或括号不匹配return false;}stack.pop(); // 是一对括号,执行出栈(消除左右括号)} else { // 左边括号,直接入栈stack.push(c);}}return stack.isEmpty();
}

我们在 LeetCode 中提交一下代码,执行结果如下:

代码解析

以上代码的 map 集合是用于定义括号的匹配规则,比如“)”对应的匹配值是“(”,“]”的匹配值是“[”等,然后我们再去循环待验证的字符串,遇到左括号直接入栈,遇到右括号让它与栈顶元素匹配,等到整个字符串循环结束,如果栈为空则说明字符串的括号合法。

复杂度分析

时间复杂度:O(n),遍历了一遍整个字符串。空间复杂度:O(n)。

实现代码二

除了使用栈之外,我们还可以使用借助 Java 中的 replace 方法来实现,我们可以循环的消除字符串中的括号,比如将“()”或“[]”或“{}”循环得替换为空,最后在执行完成之后如果字符串为空,则说明字符串中的括号是合法的,具体实现代码如下:

public boolean isValid(String s) {int len;do {len = s.length();// 消除成双成对的符号s = s.replace("()", "").replace("[]", "").replace("{}", "");} while (len != s.length()); // 不能再进行替换了,replace 方法没有替换任何字符return s.length() == 0;}

我们在 LeetCode 中提交一下代码,执行结果如下:

从运行结果来看,二者的执行效率相差还是很明显的:

总结

本文我们讲了一道 bilibili 的笔试真题,同时它也是栈的经典面试题,我们可以借助栈的特性(先进后出)将所有的左括号入栈,当遇到右括号时让它与栈顶元素进行匹配,当字符串循环结束栈为空时,则说明此字符串的括号是合法的。当然我们在实际面试中,也可以使用 Java 的 replace 方法作为一个保底的实现方案,因为 replace 方法的实现相对更简单一些,只是性能不怎么好。


往期推荐

算法图解:如何找出栈中的最小值?

链表反转的两种实现方法,后一种击败了100%的用户!

JDK 竟然是这样实现栈的?

关注下方二维码,收获更多干货!

算法图解:如何判断括号是否有效?相关推荐

  1. vb6 判断打印机是否有效_吊打面试官 | 算法之如何判断括号是否有效?

    今天要讲的这道题是 bilibili 今年的笔试真题,也是一道关于栈的经典面试题. 经过前面文章的学习,我想很多朋友已经看出来了,我接下来要写的是一个关于「算法图解」的系列文章,中间可能会穿插少量的其 ...

  2. 栈判断字符串是否为中心对称_数据结构和算法入门之判断括号字符串的合法性(valid parentheses)...

    今天终于开始看栈的部分咯!栈这个东西没啥好介绍的,我想基本只要写过一丢丢代码的人已经都非常清楚了.今天这个题目是一个非常简单但是也很经典地用到栈这个数据结构的题,废话不多说,原题链接如下: Loadi ...

  3. 《算法图解》读书笔记

    这是一本很入门的算法书,介绍的东西还算简单明了,大体补充了一些自己没理解的东西. 粗略地看了一下,感觉还是"纸上得来终觉浅,绝知此事要躬行!" <<算法图解>&g ...

  4. 算法图解——广度优先搜索

    本文章根据<算法图解>提供来写的. # -*- coding = utf-8 -*- # @Time : 2021/12/14 17:20 # @Author : 宁洪康 # @File ...

  5. 《算法图解》读书笔记—像小说一样有趣的算法入门书

    前言 学习算法课程的时候,老师推荐了两本算法和数据结构入门书,一本是<算法图解>.一本是<大话数据结构>,<算法图解>这本书最近读完了,读完的最大感受就是对算法不再 ...

  6. 《算法图解》学习笔记(十一):十种经典的算法与数据结构

    python学习之路 - 从入门到精通到大师 文章目录 [python学习之路 - 从入门到精通到大师](https://blog.csdn.net/TeFuirnever/article/detai ...

  7. 数据结构 判断括号是否匹配

    实验题目: 给定字符串,仅包含'(',')','{','}','[',']'六种字符,判断括号是否匹配 匹配规则如下 每一个左括号必须和相同类型的右括号匹配 左括号必须以正确的顺序闭合 空字符串也认为 ...

  8. 《算法图解》学习笔记(十):K 最近邻算法(附代码)

    欢迎关注WX公众号:[程序员管小亮] python学习之路 - 从入门到精通到大师 文章目录 欢迎关注WX公众号:[程序员管小亮] [python学习之路 - 从入门到精通到大师](https://b ...

  9. 算法图解1-二分法与大O表示法

    目录 一,二分法与大O表示法 1.1写在前面 1.2需要具备的知识 1.3二分法 1.4更佳的查找方式 1.4.1代码设计 1.4.2运行时间 1.5大O表示法 1.5.1算法的运行时间以不同的速度增 ...

最新文章

  1. 4天快速入门python数据挖掘_4天快速入门Python数据挖掘
  2. tensorflow 安装_tensorflow安装
  3. 使用webpack打包的后,公共请求路径的配置问题
  4. 关于系统弹出错误:429 , ActiveX 部件不能创建对象 的解决方法
  5. 这种有序神经元,像你熟知的循环神经网络吗?
  6. springboot集成spring security安全框架入门篇
  7. (转)spring中的拦截器(HandlerInterceptor+MethodInterceptor)
  8. (matlab)自定义图像(matlab)
  9. django-crontab 快速配置,高效执行django的定时任务/周期性任务
  10. go 递归tree关系_Go实现一个二叉搜索树
  11. 安装SQL Server出现在创建窗口句柄之前,不能在控件上调用 Invoke 或 BeginInvoke的错误解决办法...
  12. 数据库 : 事物以及隔离性导致的问题
  13. 无责任书评——DOOM启示录 BORLAND传奇
  14. 【ArcGIS风暴】ArcGIS10.6栅格计算器(Raster Calculator)用法详解
  15. java中string是什么意思_在java中String...是什么意思
  16. 小程序实现书籍翻页效果
  17. Java基础 常见数据结构与算法 项目总结
  18. python大漠插件多开_绝地助手_python挪用大漠插件教程04鼠键事宜及基本项目头脑...
  19. 美学心得(第二百三十五集) 罗国正
  20. 媒体查询ipad,pc端

热门文章

  1. win7无法连接打印机拒绝访问_如何解决局域网无法访问SQL Server 2008 无法连接到(local)...
  2. accept和select的区别
  3. 与专门团队一起持续交付
  4. 搞懂 Java HashMap 源码
  5. 8.4. su - root
  6. JEESZ分布式架构3--CentOs下安装MySQL(环境准备)
  7. LXD 2.0 系列(十二):调试,及给 LXD 做贡献
  8. 《Java高级程序设计》期末作业【2】-进度安排
  9. javascript Array学习与使用
  10. 第 二 十 八 天 :LB 负 载 均 衡 搭 建 之 LVS