【算法练习】leetcode算法题合集之栈和队列篇

普通栈

LeetCode20 有效的括号

LeetCode20 有效的括号

定义一个辅助map,判断字符串的字符是否在]})中。一旦是右括号就要弹出元素,判断匹配。

class Solution {
    public boolean isValid(String s) {
        if (s.length() % 2 == 1) {
            return false;
        }
        Map<Character, Character> pairs = new HashMap<>();
        pairs.put(')', '(');
        pairs.put(']', '[');
        pairs.put('}', '{');
        Stack<Character> stack = new Stack<>();
        for (int i = 0; i < s.length(); i++) {
            if (pairs.containsKey(s.charAt(i))) {
                if (stack.isEmpty() || pairs.get(s.charAt(i)) != stack.peek()) {
                    return false;
                }
                stack.pop();
            } else {
                stack.push(s.charAt(i));
            }
        }
        if (!stack.isEmpty()) {
            return false;
        }
        return true;
    }
}

LeetCode155. 最小栈

LeetCode155. 最小栈

使用栈记录最小的元素。

class MinStack {

    private Stack<Integer> stack;

    private Stack<Integer> minStack;

    public MinStack() {
        stack = new Stack<>();
        minStack = new Stack<>();
    }

    public void push(int val) {
        stack.push(val);
        if (minStack.isEmpty() || minStack.peek() >= val) {
            minStack.push(val);
        }
    }

    public void pop() {
        Integer pop = stack.pop();
        if (pop.equals(minStack.peek())) {
            minStack.pop();
        }
    }

    public int top() {
        return stack.peek();
    }

    public int getMin() {
        return minStack.peek();
    }
}

LeetCode232. 用栈实现队列

LeetCode232. 用栈实现队列

使用出栈和入栈。

出栈为空的情况下,才去调用入栈的数据。[1,2]存到入栈,出栈是[2,1],如果1这时候出去了,2保留在里面,2应该是下一个出去的元素。

class MyQueue {

    private Stack<Integer> inStack;
    private Stack<Integer> outStack;

    public MyQueue() {
        inStack = new Stack<>();
        outStack = new Stack<>();
    }

    public void push(int x) {
        inStack.push(x);
    }

    public int pop() {
        if (outStack.isEmpty()) {
            while (!inStack.isEmpty()) {
                outStack.push(inStack.pop());
            }
        }

        return outStack.pop();
    }

    public int peek() {
        if (outStack.isEmpty()) {
            while (!inStack.isEmpty()) {
                outStack.push(inStack.pop());
            }
        }
        return outStack.peek();
    }

    public boolean empty() {
        return inStack.isEmpty() && outStack.isEmpty();
    }
}

LeetCode225. 用队列实现栈

LeetCode225. 用队列实现栈

使用入队和出队,入队接收第一个元素(即将出队),把出队的元素移动到入队中,再换成出队。保证入队一直为空。

class MyStack {

    private Queue<Integer> inQueue;

    private Queue<Integer> outQueue;

    public MyStack() {
        inQueue = new LinkedList<>();
        outQueue = new LinkedList<>();
    }

    public void push(int x) {
        inQueue.offer(x);
        while (!outQueue.isEmpty()) {
            inQueue.offer(outQueue.poll());
        }
        Queue<Integer> tmp = inQueue;
        inQueue = outQueue;
        outQueue = tmp;

    }

    public int pop() {

        return outQueue.poll();
    }

    public int top() {
        return outQueue.peek();
    }

    public boolean empty() {
        return outQueue.isEmpty();
    }
}

一个队列。把原先队列中的元素添加到新来的元素后面。

class MyStack {

    private Queue<Integer> queue;

    public MyStack() {
        queue = new LinkedList<>();
    }

    public void push(int x) {
        int size = queue.size();
        queue.offer(x);
        for (int i = 0; i < size; i++) {
            queue.offer(queue.poll());
        }
    }

    public int pop() {
        return queue.poll();
    }

    public int top() {
        return queue.peek();
    }

    public boolean empty() {
        return queue.isEmpty();
    }
}

LeetCode394. 字符串解码(**)

LeetCode394. 字符串解码

字符串的拼接最好使用队列,将元素从集合中获取到用的是栈(获取后面的元素)

队列中有b,c,从栈中获取到的先是c,后是b,这是栈,会把字符串的顺序给弄反。

队列中存在jk,b,c,应该是jkbc拼接的,但是从栈中取元素是c,b,jk,需要先将集合的元素给弄反,再去拼接。

linkedList,addLast和push操作,是不一样的。

class Solution {
    public String decodeString(String s) {
        LinkedList<String> linkedList = new LinkedList<>();
        int index = 0;
        while (index < s.length()) {
            char c = s.charAt(index);
            if (Character.isDigit(c)) {
                StringBuilder sb = new StringBuilder();
                while (Character.isDigit(s.charAt(index))) {
                    sb.append(s.charAt(index));
                    index++;
                }
                linkedList.add(sb.toString());
            } else if (Character.isLetter(c) || c == '[') {
                linkedList.add(String.valueOf(c));
                index++;
            } else {
                index++;
                List<String> list = new ArrayList<>();
                // bc --> 先加入c,后添加b
                while (!linkedList.peekLast().equals("[")) {
                    list.add(linkedList.pollLast());
                }
                Collections.reverse(list);
                String string = list.stream().collect(Collectors.joining());
                linkedList.pollLast();
                int repeatTime = Integer.parseInt(linkedList.pollLast());
                StringBuilder res = new StringBuilder();
                for (int i = 0; i < repeatTime; i++) {
                    res.append(string);
                }
                linkedList.add(res.toString());
            }
        }
        StringBuilder res = new StringBuilder();
        while (!linkedList.isEmpty()) {
            res.append(linkedList.poll());
        }
        return res.toString();

    }
}

单调栈

LeetCode739. 每日温度

栈中的元素是单调递减的,找到一个大的元素就将栈中的元素弹出

26,24,22…遇到25的时候就可以记录24和22的温度在几天后有升高。

class Solution {
    public int[] dailyTemperatures(int[] temperatures) {
        int[] res = new int[temperatures.length];
        Stack<Integer> stack = new Stack<>();
        stack.push(0);
        // 26,24...递减的结构

        for (int i = 1; i < temperatures.length; i++) {
            if (temperatures[stack.peek()] < temperatures[i]) {
                while (!stack.isEmpty() && temperatures[stack.peek()] < temperatures[i]) {
                    Integer pop = stack.pop();
                    res[pop] = i - pop;
                }
                stack.push(i);
            } else {
                stack.push(i);
            }
        }
        return res;
    }
}

LeetCode42. 接雨水

LeetCode42. 接雨水

单调递减,找到一个大的元素就弹出计算结果。

class Solution {
    public int trap(int[] height) {
        Stack<Integer> stack = new Stack<>();
        int ans = 0;
        for (int i = 0; i < height.length; i++) {
            while (!stack.isEmpty() && height[stack.peek()] < height[i]) {
                Integer pop = stack.pop();
                if (!stack.isEmpty()) {
                    Integer left = stack.peek();
                    ans += (i - left - 1) * (Math.min(height[left], height[i]) - height[pop]);
                }
            }
            stack.push(i);
        }
        return ans;

    }
}

LeetCode84. 柱状图中最大的矩形(*)

LeetCode84. 柱状图中最大的矩形https://leetcode-cn.com/problems/maximal-rectangle/)

单调递增的栈,4,5,6,(3),当3进行比较的时候,就获取6,以6位底,6和3所在的距离为1,那么面积就是6。以5位底,5和3的距离是2,则结果是10.

class Solution {
    public int largestRectangleArea(int[] heights) {
        int[] new_heights = new int[heights.length + 2];
        for (int i = 1; i < heights.length + 1; i++) {
            new_heights[i] = heights[i - 1];
        }
        //存放的元素到小到大,4,5,6,(3)
        Stack<Integer> stack = new Stack<>();
        int ans = 0;
        for (int i = 0; i < new_heights.length; i++) {
            if (!stack.isEmpty() && new_heights[stack.peek()] > new_heights[i]) {
                Integer pop = stack.pop();

                ans =Math.max((i-1-stack.peek())*new_heights[pop],ans);
            }
            stack.push(i);

        }
        return ans;
    }


}

优化后的写法

对于index=0前面是没有元素的,所以stack.isEmpty() ? -1 : stack.peek()

对于index==heights.length-1后面是没有元素的,所以要计算前面的元素。

class Solution {
    public int largestRectangleArea(int[] heights) {
        Stack<Integer> stack = new Stack<>();
        int ans = 0;
        for (int i = 0; i <= heights.length; i++) {
            while (!stack.isEmpty() && (i == heights.length || heights[stack.peek()] > heights[i])) {
                Integer pop = stack.pop();
                Integer left = stack.isEmpty() ? -1 : stack.peek();
                ans = Math.max((i - 1 - left) * heights[pop], ans);
            }
            stack.push(i);
        }

        return ans;
    }
}

LeetCode85. 最大矩形

LeetCode85. 最大矩形

当j=0的时候,height数组记录第0行的数据,进行求最大面积;当j=1的时候,height的数据就是第0行的高度,更新之后就会变成第一行的数据。

求最大面积就是要判断边界。

class Solution {
    public int maximalRectangle(char[][] matrix) {
        int maxArea = 0;
        int[] heights = new int[matrix[0].length];
        for (int i = 0; i < matrix.length; i++) {
            for (int j = 0; j < matrix[0].length; j++) {
                if (matrix[i][j] == '1') {
                    heights[j] += 1;
                } else {
                    heights[j] = 0;
                }
            }
            maxArea = Math.max(maxArea, getLargestArea(heights));

        }
        return maxArea;
    }

    private int getLargestArea(int[] heights) {
        int[] new_heights = new int[heights.length + 2];
        for (int i = 1; i < heights.length + 1; i++) {
            new_heights[i] = heights[i - 1];
        }
        int ans = 0;
        // 5,7,6,3
        Stack<Integer> stack = new Stack<>();
        for (int i = 0; i < new_heights.length; i++) {
            while (!stack.isEmpty() && new_heights[i] < new_heights[stack.peek()]) {
                Integer pop = stack.pop();
                if (!stack.isEmpty()) {
                    ans = Math.max(ans, (i - stack.peek() - 1) * new_heights[pop]);
                }
            }
            stack.push(i);
        }
        return ans;
    }
}

单调队列

Leetcode239. 滑动窗口最大值

Leetcode239. 滑动窗口最大值

核心思路就是定义双端队列,可以从前面删除元素,也可以从后面删除元素。

当首位元素超过i-k了,不在范围了,进行移除。

当新增元素比末尾元素大的时候,把末尾元素移除,把新增的元素给添加上。

如此,保证队列中存在最多k个元素,且队列前端是最大的元素。其次,第二个元素是除了第一个元素最大的元素。有点像单调栈,比单调栈多了一个移除过期的元素。

class Solution {
    public int[] maxSlidingWindow(int[] nums, int k) {
        int[] res = new int[nums.length - k + 1];
        LinkedList<Integer> queue = new LinkedList<>();
        for (int i = 0; i < nums.length; i++) {
            // 0,1,2,3,4,5 当4进来的时候,k=3,小于1的数据就得清空
            while (!queue.isEmpty() && queue.peek() <= i - k) {
                queue.poll();
            }
            while (!queue.isEmpty() && nums[i] > nums[queue.peekLast()]) {
                queue.pollLast();
            }
            queue.add(i);
            if (i >= k - 1) {
                res[i - k + 1] = nums[queue.peek()];
            }
        }
        return res;
    }
}

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值