LeetCode42. 接雨水接雨水_单调栈

什么时候用单调栈呢?

    通常是一维数组,要寻找任一个元素的右边第一个比自己大或者左边第一个比自己小的元素的位置,此时我们就要想到可以用单调栈了。

    时间复杂度为$O(n)$。

单调栈的本质

    空间换时间,因为在遍历的过程中需要用一个栈来记录右边第一个比当前元素大的元素

    优点是只需要遍历一次。

在使用单调栈的时候首先要明确如下几点:

    单调栈里存放的元素是什么?

    单调栈里只需要存放元素的下标i就可以了,如果需要使用对应的元素,直接T[i]就可以获取。

单调栈里元素是递增

    注意一下顺序为 从栈头到栈底的顺序,因为单纯的说从左到右或者从前到后,不说栈头朝哪个方向的话,大家一定会越看越懵。

    这里我们要使用递增循序(再强调一下是指从栈头到栈底的顺序),因为只有递增的时候,加入一个元素i,才知道栈顶元素在数组中右面第一个比栈顶元素大的元素是i。

    文字描述理解起来有点费劲,接下来我画了一系列的图,来讲解单调栈的工作过程。

    使用单调栈主要有三个判断条件。

        当前遍历的元素T[i]小于栈顶元素T[st.top()]的情况

        当前遍历的元素T[i]等于栈顶元素T[st.top()]的情况

        当前遍历的元素T[i]大于栈顶元素T[st.top()]的情况

 

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