昨天算是简单的了解了一下栈与队列的基本操作与写法,今天开始将它应用到题目当中去。
开始做题!
20.有效的括号(简单)
给定一个只包括 '('
,')'
,'{'
,'}'
,'['
,']'
的字符串 s
,判断字符串是否有效。
有效字符串需满足:
- 左括号必须用相同类型的右括号闭合。
- 左括号必须以正确的顺序闭合。
- 每个右括号都有一个对应的相同类型的左括号。
示例 1:
输入:s = "()" 输出:true
思路:
因为用到查找,我们就得想到哈希算法。本题中左右括号可以配对,所以我们可以用map
首先得明确返回false的话有多少个条件。
继而我们可以用栈存储右括号,碰到左括号时判断相对应的右括号是否匹配。
碰到右括号就把他存进栈中。
代码实现:
class Solution {
public boolean isValid(String s) {
//如果括号数量是奇数,那一定配对不成功
int n = s.length();
if (n % 2 == 1) {
return false;
}
Map<Character, Character> pairs = new HashMap<Character, Character>() {{
put(')', '(');
put(']', '[');
put('}', '{');
}};
Deque<Character> stack = new LinkedList<Character>();
for (int i = 0; i < n; i++) {
char ch = s.charAt(i);
if (pairs.containsKey(ch)) {
//若栈中无右括号或者栈中最后存储的元素与本左括号不匹配,错误
if (stack.isEmpty() || stack.peek() != pairs.get(ch)) {
return false;
}
stack.pop();
} else {
stack.push(ch);
}
}
return stack.isEmpty();
}
}
作者:力扣官方题解
链接:https://leetcode.cn/problems/valid-parentheses/solutions/373578/you-xiao-de-gua-hao-by-leetcode-solution/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
总结:学会栈的建立方法及思想。
1047.删除字符串中的所有相邻重复项
给出由小写字母组成的字符串 S
,重复项删除操作会选择两个相邻且相同的字母,并删除它们。
在 S 上反复执行重复项删除操作,直到无法继续删除。
在完成所有重复项删除操作后返回最终的字符串。答案保证唯一。
示例:
输入:"abbaca" 输出:"ca"
思路:通过上题我们可以得到经验,本题也可以用栈来解决。
代码实现:
import java.util.Deque;
import java.util.LinkedList;
class Solution {
public String removeDuplicates(String s) {
Deque<Character> stack = new LinkedList<Character>();
for (int i = 0; i < s.length(); i++) {
char x = s.charAt(i);
if(stack.isEmpty()||stack.peek()!=x){
stack.push(x);
}
else stack.pop();
}
//下面是将栈转化为字符串
String ss = "";
while (!stack.isEmpty()){
ss=stack.pop()+ss;
}
return ss;
}
}
总结:本题中,应该学会如何将栈中内容转化为字符串并输出。注意:倒数第二行中stack.pop()与ss顺序不可颠倒。
150.逆波兰表达式求值(中等)
给你一个字符串数组 tokens
,表示一个根据 逆波兰表示法 表示的算术表达式。
请你计算该表达式。返回一个表示表达式值的整数。
注意:
- 有效的算符为
'+'
、'-'
、'*'
和'/'
。 - 每个操作数(运算对象)都可以是一个整数或者另一个表达式。
- 两个整数之间的除法总是 向零截断 。
- 表达式中不含除零运算。
- 输入是一个根据逆波兰表示法表示的算术表达式。
- 答案及所有中间计算结果可以用 32 位 整数表示。
示例 1:
输入:tokens = ["2","1","+","3","*"] 输出:9
思路:
逆波兰表达式:是一种后缀表达式,所谓后缀就是指运算符写在后面。
平常使用的算式则是一种中缀表达式,如 ( 1 + 2 ) * ( 3 + 4 ) 。
该算式的逆波兰表达式写法为 ( ( 1 2 + ) ( 3 4 + ) * ) 。
本题用栈的思想解决的话,应该还是很好理解的。但具体代码怎么实现,还是要多家练习。
代码实现:
class Solution {
public int evalRPN(String[] tokens) {
Deque<Integer> stack = new LinkedList();
for (String s : tokens) {
if ("+".equals(s)) { // leetcode 内置jdk的问题,不能使用==判断字符串是否相等
stack.push(stack.pop() + stack.pop()); // 注意 - 和/ 需要特殊处理
} else if ("-".equals(s)) {
stack.push(-stack.pop() + stack.pop());
} else if ("*".equals(s)) {
stack.push(stack.pop() * stack.pop());
} else if ("/".equals(s)) {
int temp1 = stack.pop();
int temp2 = stack.pop();
stack.push(temp2 / temp1);
} else {
stack.push(Integer.valueOf(s));
}
}
return stack.pop();
}
}