今天要讲的这道题是 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) { // 括号不是成对出现直接返回 false        return false;    }    // 把所有对比的括号存入 map,对比时用    Map map = new HashMap<>();    map.put(')', '(');    map.put('}', '{');    map.put(']', '[');// 定义栈,用于存取括号(辅助比较)    Stack stack = new Stack<>();for (int i = 0; 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 方法的实现相对更简单一些,只是性能不怎么好。

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

  1. 重复订单号校验_吊打面试官系列重复消费、顺序消费、分布式事务

    你知道的越多,你不知道的越多 前言 消息队列在互联网技术存储方面使用如此广泛,几乎所有的后端技术面试官都要在消息队列的使用和原理方面对小伙伴们进行360°的刁难. 作为一个在互联网公司面一次拿一次Of ...

  2. atoi函数_吊打面试官 | 腾讯经典考点写代码实现atoi函数

    点击蓝字关注我哦 以下是本期干货视频视频后还附有文字版本哦 ▼<腾讯经典考点-写代码实现atoi函数>▼ ps:请在WiFi环境下打开,如果有钱任性请随意 在腾讯面试时,经常会被问到如何用 ...

  3. 如何查找历史线程阻塞原因_吊打面试官!Java多线程并发 108 道题,你能答对多少?...

    多线程并发108题 1.Java中实现多线程有几种方法 2.继承Thread 类 3.实现Runnable 接口. 4.ExecutorService. Callable. Future 有返回值线程 ...

  4. c调用其他类的方法_吊打面试官-类加载器

    1. 什么是类加载器? 类加载器(class loader)用来加载 Java 类到 Java 虚拟机中.一般来说,Java 类的虚拟机使用 Java 方式如下:Java 源程序(.java 文件)在 ...

  5. mabatisplus怎么给实体类自定义属性_吊打面试官之:当实体类中的属性名和表中的字段名不一样 ,怎么办 ?...

    第1种: 通过在查询的sql语句中定义字段名的别名,让字段名的别名和实体类的属性名一致. <select id="selectorder" parametertype=&qu ...

  6. qt定时器是阻塞的吗_吊打面试官 | 面试官:TCP真的可靠吗

    点击蓝字关注我哦 以下是本期干货视频视频后还附有文字版本哦 ▼<面试官:TCP真的可靠吗>▼ ps:请在WiFi环境下打开,如果有钱任性请随意 TCP真的可靠吗 面试官经常会问的一个问题是 ...

  7. redis 亿级查询速度_吊打面试官系列:Redis 性能优化的 13 条军规大全

    我的官方群点击此处. 1.缩短键值对的存储长度 键值对的长度是和性能成反比的,比如我们来做一组写入数据的性能测试,执行结果如下: 从以上数据可以看出,在 key 不变的情况下,value 值越大操作效 ...

  8. mysql 触发器条件判断偶尔失效_mysql┃多个角度说明sql优化,让你吊打面试官!...

    ​正文约: 4744字 预计阅读时间: 12分钟 文章首发于我的微信公众号:moon聊技术,欢迎大家关注 mysql┃多个角度说明sql优化,让你吊打面试官! 目录 目录 前言 正文 1.表结构优化● ...

  9. 自动化测试面试题及答案,看完后吊打面试官!

    自动化测试是什么?自动化测试学什么?自动化测试面试题及答案?–看完后吊打面试官! 一.前言 最近有童鞋和我抱怨,说网上很难搜到那些全面又合适的自动化测试面试题,这里根据我个人的经验以及收集整理的: 你 ...

最新文章

  1. python selenium xpath_python+selenium十四:xpath和contains模糊匹配
  2. 科学:螳螂虾大脑拥有记忆和学习中心
  3. NR 5G QoS模型
  4. 自学python还是报班-转行Python开发自学还是报班?老男孩全日制学习
  5. 实地址模式与保护模式下的中断与异常处理
  6. h5py快速入门指南
  7. Yii中的CComponent应用实例
  8. pypthon3精要(11)-try,except,else异常处理
  9. Bourbon: 让你的sass更简洁
  10. Java 高级—— IO 基础
  11. yii2 mysql查询_Yii2 数据库查询汇总
  12. JavaScript常用算法
  13. vim 批量替换字符串_Vim 有什么奇技淫巧?
  14. LINUX 添加xp虚拟机
  15. linux rootkit扫描,如何扫描我的Linux系统的rootkit,蠕虫,木马等?
  16. Android性能优化—TraceView的使用
  17. C++ 解决大数运算(大数加法,大数幂运算,大数求余)
  18. JS中什么是回调函数?
  19. 吴裕雄--天生自然 诗经:临江仙·滚滚长江东逝水
  20. PHP中explode和implode的区别

热门文章

  1. 漫反射 高光反射_如何有效地使用反射
  2. JUnit5 TestSuite替代
  3. docker集群_使用Docker,Chef和Amazon OpsWorks进行集群范围的Java / Scala应用程序部署...
  4. mapreduce 算法_MapReduce算法–了解数据联接第二部分
  5. 宣布EAXY:使Java中的XML更容易
  6. Java 10:“ var”关键字
  7. 编译hotspot_从Hotspot JIT编译器打印生成的汇编代码
  8. 消息队列概述[幻灯片]
  9. Java开发人员必须看到的13个Decks保持更新
  10. spring 事件模型_Spring–设计领域模型和服务层