代码随想录第11天 | LeetCode20. 有效的括号、LeetCode1047. 删除字符串中的所有相邻重复项、LeetCode150. 逆波兰表达式求值

day11 | LeetCode20. 有效的括号、LeetCode1047. 删除字符串中的所有相邻重复项、LeetCode150. 逆波兰表达式求值

一、题目:LeetCode20. 有效的括号

链接:20. 有效的括号 - 力扣(LeetCode)

方法:使用栈匹配解决

1. 思路:

在做题之前先分析,可能出现的匹配情况:

  1. { [ ( ] ) ] } :字符串括号右方向多余
  2. { [ ( ] ] } :字符串括号没有多余,但是括号类型没有匹配
  3. { [ ( ( ) ] } :字符串括号左方向多余

这里有个技巧,在匹配左括号时,进行右括号入栈,就只需要比较当前元素和栈顶相不相等就可以了。

2. 代码实现:

class Solution {
    public boolean isValid(String s) {
        Deque<Character> deque = new LinkedList<Character>();
        char ch;
        for (int i = 0; i < s.length(); i++) {
            ch = s.charAt(i);
            if (ch == '(') {
                deque.push(')');
            } else if (ch == '[') {
                deque.push(']');
            } else if (ch == '{') {
                deque.push('}');
                // 遇到不匹配的情况,跳出
            } else if (deque.isEmpty() || deque.peek() != ch) {
                return false;
                // 括号匹配消除
            } else {
                deque.pop();
            }
        }
        // 判断栈是否为空,空代表括号全部匹配
        return deque.isEmpty();
    }
}

3.复杂度分析:

  • 时间复杂度:O(N)
  • 空间复杂度:O(N)

Reference

  1. 代码随想录 (programmercarl.com)
  2. 栈的拿手好戏!| LeetCode:20. 有效的括号_哔哩哔哩_bilibili

二、题目:LeetCode1047. 删除字符串中的所有相邻重复项

链接:1047. 删除字符串中的所有相邻重复项 - 力扣(LeetCode)

方法:使用栈匹配解决

1. 思路:

在做题之前先分析,可能出现的匹配情况:

  1. { [ ( ] ) ] } :字符串括号右方向多余
  2. { [ ( ] ] } :字符串括号没有多余,但是括号类型没有匹配
  3. { [ ( ( ) ] } :字符串括号左方向多余

这里有个技巧,在匹配左括号时,进行右括号入栈,就只需要比较当前元素和栈顶相不相等就可以了。

2. 代码实现:

使用 ArrayDeque 作为堆栈:

class Solution {
    public String removeDuplicates(String s) {
        // ArrayDeque会比LinkedList在除了删除元素这一点外会快一点
        ArrayDeque<Character> deque = new ArrayDeque<>();
        char ch;
        for (int i = 0; i < s.length(); i++) {
            ch = s.charAt(i);
            // 不匹配放入栈中
            if (deque.isEmpty() || deque.peek() != ch){
                deque.push(ch);
            // 匹配就消除,对对碰
            } else {
                deque.pop();
            }
        }
        String str = "";
        // 剩余的元素即为不重复的元素
        while (!deque.isEmpty()) {
            str = deque.pop() + str;
        }
        return str;
    }
}

直接使用字符串来当栈,省去栈转字符串的操作:

class Solution {
    public String removeDuplicates(String s) {
        // 将字符串当做栈
        StringBuffer res = new StringBuffer();
        int top = -1;
        for (int i = 0; i < s.length(); i++) {
            char ch = s.charAt(i);
            // 栈不为空并且和当前字符匹配
            if (top >= 0 && res.charAt(top) == ch){
                res.deleteCharAt(top);
                top--;
            // 不匹配存入栈
            } else {
                res.append(ch);
                top++;
            }
        }
        return res.toString();
    }
}

3.复杂度分析:

  • 时间复杂度:O(N)
  • 空间复杂度:O(N)

Reference

  1. 代码随想录 (programmercarl.com)
  2. 栈的好戏还要继续!| LeetCode:1047. 删除字符串中的所有相邻重复项_哔哩哔哩_bilibili

三、题目:LeetCode150. 逆波兰表达式求值

链接:150. 逆波兰表达式求值 - 力扣(LeetCode)

逆波兰表达式:是一种后缀表达式,所谓后缀就是指算符写在后面。

平常使用的算式则是一种中缀表达式,如: ( 1 + 2 ) ∗ ( 3 + 4 ) ( 1 + 2 ) * ( 3 + 4 ) (1+2)(3+4)

该算式的逆波兰表达式写法为: ( ( 12 + ) ( 34 + ) ∗ ) ( ( 1 2 + ) ( 3 4 + ) * ) ((12+)(34+))

逆波兰表达式主要有以下两个优点:

  • 去掉括号后表达式无歧义,上式即便写成 12 + 34 + ∗ 1 2 + 3 4 + * 12+34+ 也可以依据次序计算出正确结果。
  • 适合用栈操作运算:遇到数字则入栈;遇到算符则取出栈顶两个数字进行计算,并将结果压入栈中。

方法:使用栈解决

1. 思路:

本题的本质就是,每一个子表达式要得出结果,然后再拿这个结果进行运算。其实就是一个相邻字符串消除的过程。

2. 代码实现:

class Solution {
    public int evalRPN(String[] tokens) {
        Deque<Integer> stack = new LinkedList<>();
        for (String token : tokens) {
            if (token.equals("+") || token.equals("-") || token.equals("*") || token.equals("/")) {
                int num1 = stack.pop();
                int num2 = stack.pop();

                // 判断表达式
                switch (token) {
                    case "+":
                        stack.push(num1 + num2);
                        break;
                    case "-":
                        stack.push(num2 - num1);
                        break;
                    case "*":
                        stack.push(num1 * num2);
                        break;
                    case "/":
                        stack.push(num2 / num1);
                        break;
                }
            } else {
                stack.push(Integer.valueOf(token));
            }
        }
        return stack.pop();
    }
}

3. 复杂度分析:

  • 时间复杂度:O(N)
  • 空间复杂度:O(N)

思考:

  1. 使用**后缀表达式对计算机来说非常的友好。**计算机只需要从左往右依次的往下扫描,而不需要判断表达式的优先级。

Reference

  1. 代码随想录 (programmercarl.com)
  2. 栈的最后表演! | LeetCode:150. 逆波兰表达式求值_哔哩哔哩_bilibili
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值