前言
栈和队列篇,继续。
记录 二十九【20. 有效的括号】
一、题目阅读
给定一个只包括 ‘(’,‘)’,‘{’,‘}’,‘[’,‘]’ 的字符串 s ,判断字符串是否有效。
有效字符串需满足:
左括号必须用相同类型的右括号闭合。
左括号必须以正确的顺序闭合。
每个右括号都有一个对应的相同类型的左括号。
示例 1:
输入:s = "()"
输出:true
示例 2:
输入:s = "()[]{}"
输出:true
示例 3:
输入:s = "(]"
输出:false
提示:
1 <= s.length <= 104
s 仅由括号 '()[]{}' 组成
二、尝试实现
思路
题目意思是看括号有没有成功配对,并且配对顺序由内层到外层合理。
(1)首先,如果string长度是奇数,肯定是false。
(2)双指针法试试:行不通。
示例一:s = " {,[,(,),],} ",从两端指或者从一半的地方指;
对于示例二:s = "()[]{}",不成立,所以pass。
(3)只能从头遍历string s.
- 如果遇到左括号,压入栈里;
- 如果遇到右括号,取一个栈顶元素,看是否配对?用栈可以满足大括号套中括号套小括号,嵌套时候的顺序配对正确。
代码实现
class Solution {
public:
bool isValid(string s) {
if(s.size()%2 != 0){ //奇数,肯定不配对。
return false;
}
stack<char> stack1;
for(int i = 0;i < s.size();i++){
if(s[i] == '(' || s[i] == '[' || s[i] == '{'){
stack1.push(s[i]);
}else if(s[i] == ')'){
if(!stack1.empty() && stack1.top() == '('){ //补充添加stack1不为空,如果有括号先出现,访问了空栈。
stack1.pop();
}else{
return false;
}
}else if(s[i] == ']'){
if(!stack1.empty() && stack1.top() == '['){
stack1.pop();
}else{
return false;
}
}else if(s[i] == '}'){
if(!stack1.empty() && stack1.top() == '{'){
stack1.pop();
}else{
return false;
}
}
}
if(stack1.empty()){ //整个for循环结束,成功配对,stack1中不应该有元素。
return true;
}else{
return false;
}
}
};
两处细节修正:
- 如果s只有左括号,也是不配对的;for循环判断不出,所以最后加if(stack1.empty())。成功配对后,stack1中肯定是空。
- 如果一上来就是右括号,只是if(stack1.top() == ‘{’),出现 访问空栈;所以需要加if(!stack1.empty() && stack1.top() == ‘(’) ,是否为空的判断。
三、参考思路
学习内容
- 栈在对称匹配类题目中作用很大。
- 先分析了三种不匹配的情况: 左括号多余;右括号多余;括号类型不匹配;
- 技巧:如果是s[i]左括号,不是把左括号本身放到栈里,而是把其对应的右括号放到栈里。当s[i]是右括号时:只需要看当前栈顶元素是不是相等。
- 如果栈为空:说明右括号多余;
- 如果栈顶不相等:说明嵌套的顺序不对;
- 如果遍历结束,栈不为空:说明左括号多余。
- 总结:这个思路一开始就把所有情况都囊括,无需在实现之后查漏补缺。
- 对比思路下来:发现把右括号放进去,代码更简练,而不是拿着右括号去找左括号。
代码实现
class Solution {
public:
bool isValid(string s) {
if(s.size()%2 != 0){ //奇数,肯定不配对。
return false;
}
stack<char> stack1;
for(int i = 0;i < s.size();i++){
if(s[i] == '(')
stack1.push(')');
else if(s[i] == '[')
stack1.push(']');
else if(s[i] == '{')
stack1.push('}');
else if(stack1.empty() || stack1.top() != s[i]){
return false;
}
else stack1.pop();//注意:这里还得有else
}
return stack1.empty();
}
};
有几处写的很好:
- 这样for循环return false是一条语句。
- 最后不是if判断空,而是直接return stack1.empty();
总结
- 栈用来解决对称匹配问题。
- 把匹配方(右括号)放进去,而不是把自己(左括号)放进去。拿着匹配方(右括号)去找相不相等。
(欢迎指正,转载标明出处)