每日一道 LeetCode (6):有效的括号

每天 3 分钟,走上算法的逆袭之路。
前文合集
代码仓库
GitHub: https://github.com/meteor1993/LeetCode
Gitee: https://gitee.com/inwsy/LeetCode
题目:有效的括号
题目来源:https://leetcode-cn.com/problems/valid-parentheses/
给定一个只包括 '(',')','{','}','[',']' 的字符串,判断字符串是否有效。
有效字符串需满足:
- 左括号必须用相同类型的右括号闭合。
- 左括号必须以正确的顺序闭合。
注意空字符串可被认为是有效字符串。
示例 1:
输入: "()"
输出: true
示例 2:
输入: "()[]{}"
输出: true
示例 3:
输入: "(]"
输出: false
示例 4:
输入: "([)]"
输出: false
示例 5:
输入: "{[]}"
输出: true
解题思路
解题思路?
啊呸,有个 P 的解题思路,我跟你们讲哦,这种题没见过,你就是想不出来。
我自己思索了几分钟以后,果断投降去看答案了,这玩意,不是我这种新手小白搞的定的。
结果,一看答案秒懂。

整个解题思路是借用了数据结构「栈」的「先进后出」的特性。
首先第一件事情还是先思考清楚极限情况,比如如果是空字符串,是可以返回 true ,如果字符串长度是奇数,那么显然是无法满足左右括号的对应关系。
接下来就是核心问题,一个左括号一个右括号,中间还可能隔着千山万水,如何处理?
使用「栈」结构:
- 遇到左括号就压入栈。
- 遇到右括号就和栈顶的左括号比对,匹配失败直接返回 false 。
- 如果匹配成功,则可能是嵌套在其它匹配括号中的,所以此时要将当前栈顶的左括号弹出。
- 如果最后最终,栈中没有剩余元素,也就是没有剩下左括号,说明刚好完成匹配,括号字符串有效。否则匹配失败,括号字符串无效。
如果上面这一段不好理解,可以借助下面这个动图(来源:LeetCode):

代码实现
有了上面的思路,写代码都是小事儿了,先看一个我自己写的,完全符合上面的逻辑:
public boolean isValid(String s) {
if (s.length() == 0) return true;
if (s.length() % 2 == 1) return false;
Stack<Character> stack = new Stack<> ();
for (int i = 0; i < s.length(); i++) {
char charAt = s.charAt(i);
// 如果是左括号,则把字符压入栈
if (charAt == '(' || charAt == '{' || charAt == '[')
stack.push(charAt);
else {
// 如果此时还有右括号而栈中已无左括号
if (stack.isEmpty()) return false;
// 获取栈顶的值
char top = stack.peek();
// 如果栈顶的值等于右括号,则出栈,否则返回 false
if ((top == '{' && charAt == '}') || (top == '(' && charAt == ')') || (top == '[' && charAt == ']'))
stack.pop();
else
return false;
}
}
return stack.isEmpty();
}

里面的注释已经比较清晰了,我就不多解释了。
接着我看了官方提供的答案,基本上思路和我的代码保持一致,只是把左右括号放入到了哈希表中,由哈希表来判断括号是否存在。
private Map<Character, Character> map;
// 初始化哈希表
public Solution() {
this.map = new HashMap<> ();
this.map.put(')', '(');
this.map.put('}', '{');
this.map.put(']', '[');
}
public boolean isValid_1(String s) {
if (s.length() == 0) return true;
if (s.length() % 2 == 1) return false;
Stack<Character> stack = new Stack<> ();
for (int i = 0; i < s.length(); i++) {
char charAt = s.charAt(i);
// 如果不是右括号,则把字符压入栈
if (!this.map.containsKey(charAt)) {
stack.push(charAt);
} else {
// 如果此时还有右括号而栈中已无左括号
if (stack.isEmpty()) return false;
// 获取栈顶的值
char top = stack.peek();
// 如果栈顶的值等于右括号,则出栈,否则返回 false
if (top == this.map.get(charAt))
stack.pop();
else
return false;
}
}
return stack.empty();
}

因为哈希表中的数据太少,而且寻址的次数也不够多,所以哈希表的方案看起来还比顺次匹配的的方案耗时高。
效率更高的方案有没有,当然有,直接使用数组模拟栈的入栈和出栈,这个时间是可以压缩在 1ms 以内的,执行时间可以在 Java 的提交中击败 100% 的用户,这个示例我就不写了,感兴趣的同学可以自己写写看。
在我翻答案的时候看到一个 1ms 的方案,这个方案是对我前面的那个 2ms 的代码的优化,有点意思,拿出来聊一下:
public boolean isValid_2(String s) {
char[] chs = s.toCharArray();
Stack<Character> stack = new Stack<>();
for (char c : chs) {
if (c == '{') {
stack.push('}');
} else if (c == '[') {
stack.push(']');
} else if (c == '(') {
stack.push(')');
} else if (stack.isEmpty() || stack.pop() != c) {
return false;
}
}
return stack.isEmpty();
}

这个方案就是在遇到左括号的时候往栈里面放一个对应的右括号,这么做的原因我猜测是为了好判断,只需要取出的时候和当前循环的右括号做一次 == 判断,如果相等就继续循环,不相等就直接返回 false 了。
不得不说这个方案的构思很巧妙,在已有的方案上做了相当极致的优化,把耗时进一步降低。果然姜还是老的辣,你大爷还是你大爷。

果然是学无止境,希望各位刷 LeetCode 的同学,如果时间充足,也可以多看看不同的答案,对开阔思路和视野真的是相当有帮助。
顺便吐槽下:想到这些方案的大神太 TM 变态了,这个脑回路和常人差的太远了,能用数组把耗时压到 1ms 以内我能接受,但是能把栈这种数据结构的方案优化到 2ms 以内,这个我是真的服。
每日一道 LeetCode (6):有效的括号的更多相关文章
- 每日一道 LeetCode (3):回文数
前文合集 每日一道 LeetCode 文章合集 题目:回文数 题目来源:https://leetcode-cn.com/problems/palindrome-number/ 判断一个整数是否是回文数 ...
- 每日一道 LeetCode (5):最长公共前缀
前文合集 每日一道 LeetCode 前文合集 代码仓库 GitHub: https://github.com/meteor1993/LeetCode Gitee: https://gitee.com ...
- 每日一道 LeetCode (8):删除排序数组中的重复项和移除元素
每天 3 分钟,走上算法的逆袭之路. 前文合集 每日一道 LeetCode 前文合集 代码仓库 GitHub: https://github.com/meteor1993/LeetCode Gitee ...
- 每日一道 LeetCode (9):实现 strStr()
每天 3 分钟,走上算法的逆袭之路. 前文合集 每日一道 LeetCode 前文合集 代码仓库 GitHub: https://github.com/meteor1993/LeetCode Gitee ...
- 每日一道 LeetCode (10):搜索插入位置
每天 3 分钟,走上算法的逆袭之路. 前文合集 每日一道 LeetCode 前文合集 代码仓库 GitHub: https://github.com/meteor1993/LeetCode Gitee ...
- 每日一道 LeetCode (14):数组加一
每天 3 分钟,走上算法的逆袭之路. 前文合集 每日一道 LeetCode 前文合集 代码仓库 GitHub: https://github.com/meteor1993/LeetCode Gitee ...
- 每日一道 LeetCode (15):二进制求和
每天 3 分钟,走上算法的逆袭之路. 前文合集 每日一道 LeetCode 前文合集 代码仓库 GitHub: https://github.com/meteor1993/LeetCode Gitee ...
- 每日一道 LeetCode (19):合并两个有序数组
每天 3 分钟,走上算法的逆袭之路. 前文合集 每日一道 LeetCode 前文合集 代码仓库 GitHub: https://github.com/meteor1993/LeetCode Gitee ...
- 每日一道 LeetCode (41):阶乘后的零
每天 3 分钟,走上算法的逆袭之路. 前文合集 每日一道 LeetCode 前文合集 代码仓库 GitHub: https://github.com/meteor1993/LeetCode Gitee ...
随机推荐
- 《SpringBoot判空处理》接开@valid的面纱
一.事有起因 我们在与前端交互的时候,一般会遇到字段格式校验及非空非null的校验,在没有SpringBoot注解的时候, 我们可能会在service进行处理: if(null == name){ t ...
- 从0开始,手把手教你使用React开发答题App
项目演示地址 项目演示地址 项目源码 项目源码 其他版本教程 Vue版本 小程序版本 项目代码结构 前言 React 框架的优雅不言而喻,组件化的编程思想使得React框架开发的项目代码简洁,易懂,但 ...
- Python3笔记027 - 6.2 参数传递
第6章 函数 6.2 参数传递 在理解形参和实参的基础上,理解位置参数.关键字参数.可变参数这三种情形,以及这三种的混合情形. 6.2.1 形式参数和实际参数 形式参数:在定义函数时,函数名后面括号中 ...
- PHP create_function()代码注入
查看代码 分析 变量$action要出现数字字母以外的字符,还要执行函数. /i不区分大小写 /s匹配任何不可见字符,包括空格.制表符.换页符等等 /D如果使用$限制结尾字符,则不允许结尾有换行 这里 ...
- 数据可视化之powerBI基础(二十)Power BI度量值和新建表,有什么异同?
https://zhuanlan.zhihu.com/p/101812525 PowerBI中,有三个地方可以使用DAX,分别是度量值.新建列和新建表,这三个功能并成一排摆放在这里,如图所示, 之前 ...
- 基于animate.css动画库的全屏滚动小插件,适用于vue.js(移动端、pc)项目
功能简介 基于animate.css动画库的全屏滚动,适用于vue.js(移动端.pc)项目. 安装 npm install vue-animate-fullpage --save 使用 main.j ...
- VTK根据三维坐标点集生成点云
一个简单的利用VTK根据三维坐标点集生成点云的例子,仅供参考. 一.环境:vtk-8.1 & vs2013(需自行配置vtk的环境) 二.我所读取的三维坐标点集为txt格式文件,每个点的x,y ...
- java中同步异步阻塞和非阻塞的区别
同步 所谓同步,就是在发出一个功能调用时,在没有得到结果之前,该调用就不返回. 按照这个定义,其实绝大多数函数都是同步调用(例如sin, isdigit等).但是一般而言,我们在说同步.异步的时候,特 ...
- OSCP Learning Notes - Exploit(2)
Compiling an Exploit Exercise: samba exploit 1. Search and download the samba exploit source code fr ...
- Java 线程与同步的性能优化
本文探讨的主题是,如何挖掘出Java线程和同步设施的最大性能. 1.线程池与ThreadPoolExecutor 1)线程池与ThreadPoolExecutor 线程池的实现可能有所不同,但基本概念 ...