6.4 单调栈单调栈是一种基于栈的数据结构,所谓的单调就是满足单调递增(单调递减)的栈。主要用于解决 next_greater
问题,也就是找到下一个更大的元素。
基本模板如下:
1 2 3 4 5 6 7 8 stack;for (int i = 0 ; i < n ; i++) { while (stack && nums[stack.top] < nums[i]) { int top = stack.pop(); } stack.push(i); }
给定一个整数数组 temperatures
,表示每天的温度,返回一个数组 answer
,其中 answer[i]
是指对于第 i 天,下一个更高温度出现在几天后。如果气温在这之后都不会升高,请在该位置用 0 来代替。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 示例 1 : 输入 : temperatures = [73,74,75,71,69,72,76,73] 输出 : [1,1,4,2,1,1,0,0] 示例 2 : 输入 : temperatures = [30,40,50,60] 输出 : [1,1,1,0] 示例 3 : 输入 : temperatures = [30,60,90] 输出 : [1,1,0] 提示: 1 <= temperatures.length <= 10530 <= temperatures[i] <= 100
题目解析
此题比较简单,属于单调栈的模板题。目的也是简单的找到一个更大的元素的位置。
代码展示
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 class Solution { public int [] dailyTemperatures(int [] temperatures) { Deque<Integer> stack = new LinkedList <>(); int [] res = new int [temperatures.length]; for (int i = 0 ; i < temperatures.length; i++) { while (!stack.isEmpty() && temperatures[stack.getLast()] < temperatures[i]) { int top = stack.pollLast(); res[top] = i - top; } stack.addLast(i); } return res; } }
nums1
中数字 x
的 下一个更大元素 是指 x
在 nums2
中对应位置 右侧 的 第一个 比 x 大的元素。
给你两个 没有重复元素 的数组 nums1
和 nums2
,下标从 0 开始计数,其中 nums1
是 nums2
的子集。
对于每个 0 <= i < nums1.length
,找出满足 nums1[i] == nums2[j]
的下标 j
,并且在 nums2
确定 nums2[j]
的 下一个更大元素 。如果不存在下一个更大元素,那么本次查询的答案是 -1
。
返回一个长度为 nums1.length
的数组 ans
作为答案,满足 ans[i]
是如上所述的 下一个更大元素 。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 示例 1 : 输入:nums1 = [4 ,1 ,2 ], nums2 = [1 ,3 ,4 ,2 ]. 输出:[-1 ,3 ,-1 ] 解释:nums1 中每个值的下一个更大元素如下所述: - 4 ,nums2 = [1 ,3 ,4 ,2 ]。不存在下一个更大元素,所以答案是 -1 。 - 1 ,nums2 = [1 ,3 ,4 ,2 ]。下一个更大元素是 3 。 - 2 ,nums2 = [1 ,3 ,4 ,2 ]。不存在下一个更大元素,所以答案是 -1 。 示例 2 : 输入:nums1 = [2 ,4 ], nums2 = [1 ,2 ,3 ,4 ]. 输出:[3 ,-1 ] 解释:nums1 中每个值的下一个更大元素如下所述: - 2 ,nums2 = [1 ,2 ,3 ,4 ]。下一个更大元素是 3 。 - 4 ,nums2 = [1 ,2 ,3 ,4 ]。不存在下一个更大元素,所以答案是 -1 。 提示:1 <= nums1.length <= nums2.length <= 1000 0 <= nums1[i], nums2[i] <= 10 ^4 nums1和nums2中所有整数 互不相同 nums1 中的所有整数同样出现在 nums2 中
题目解析
此题是找到nums1[i]对应的元素在nums2中找到下一个更大的元素,典型的单调栈模板题。大体思路如下:
统计nums2中每个元素对应的“下一个更大的元素” 遍历nums1,找到nums1中每个元素对应的答案。 代码展示
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 class Solution { public int [] nextGreaterElement(int [] nums1, int [] nums2) { Deque<Integer> stack = new LinkedList <>(); HashMap<Integer, Integer> map = new HashMap <>(); for (int i = 0 ; i < nums2.length; i++) { while (!stack.isEmpty() && nums2[stack.peekLast()] < nums2[i]) { int top = stack.pollLast(); map.put(nums2[top], nums2[i]); } stack.addLast(i); } int [] res = new int [nums1.length]; for (int i = 0 ; i < nums1.length; i++) { res[i] = map.getOrDefault(nums1[i], -1 ); } return res; } }
给定一个循环数组 nums
( nums[nums.length - 1]
的下一个元素是 nums[0]
),返回 nums
中每个元素的 下一个更大元素 。
数字 x
的 下一个更大的元素 是按数组遍历顺序,这个数字之后的第一个比它更大的数,这意味着你应该循环地搜索它的下一个更大的数。如果不存在,则输出 -1 。
1 2 3 4 5 6 7 8 9 10 11 12 13 示例 1 : 输入 : nums = [1,2,1] 输出 : [2,-1,2] 解释 : 第一个 1 的下一个更大的数是 2;数字 2 找不到下一个更大的数; 第二个 1 的下一个最大的数需要循环搜索,结果也是 2。 示例 2 : 输入 : nums = [1,2,3,4,3]输出: [2,3,4,-1,4] 提示 : 1 <= nums.length <= 10^4 -10^9 <= nums[i] <= 10^9
题目解析
此题也是单调栈的模板应用,除了一个点:循环数组。也就是我们需要对循环数组加以处理即可。
处理循环数组的思路如下:我们把数组想象成两个数组相加,如 [1,2,1],我们当成[1,2,1,1,2,1]。如此一来我们只需要对新的数组找到next_greater即可,结果应该为[2,-1,2,2,-1,-1],我们只要截取前半部分即可。实际写代码的过程并不需要真正创建两倍的数组,按照思路进行处理即可。新数组中的元素下标对应到原数组的下标应该 mod nums.length即可。
代码展示
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 class Solution { public int [] nextGreaterElements(int [] nums) { int [] res = new int [nums.length]; Arrays.fill(res, -1 ); Deque<Integer> stack = new LinkedList <>(); for (int i = 0 ; i < 2 * nums.length; i++) { while (!stack.isEmpty() && nums[stack.getLast() % nums.length] < nums[i % nums.length]) { int top = stack.pollLast(); if (top < nums.length) { res[top] = nums[i % nums.length]; } } stack.addLast(i); } return res; } }
给你一个字符串 s
,请你去除字符串中重复的字母,使得每个字母只出现一次。需保证 返回结果的字典序最小 (要求不能打乱其他字符的相对位置)。
1 2 3 4 5 6 7 8 9 10 11 示例 1 : 输入:s = “bcabc” 输出:“abc” 示例 2 : 输入:s = “cbacdcbc” 输出:“acdb” 提示:1 <= s.length <= 10 ^4 s 由小写英文字母组成
题目解析
此题属于比较有难度的单调栈优化的题目。如何找到字典序最小的字符串呢?很明显应该是尽量使得字典序单调递增。我们以样例1进行举例:我们不断往后选取字符的过程如下:
b bc【因为bc满足单调上升】 bca,此时a不满足单调上升,并且 c在a后面还有出现,因此我们可以选择删除c,因此得到的是ba 由于ba不满足单调上升并且b在a后面还有出现,因此我们可以选择删除b,因此得到a ab abc 上述就是单调栈的过程,尽量使得栈中元素符合单调递增的关系,才可以使得字典序最小。
代码展示
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 class Solution { public String removeDuplicateLetters (String s) { Deque<Integer> stack = new LinkedList <>(); int [] cnter = new int [26 ]; char [] cs = s.toCharArray(); for (char c : cs) { cnter[c - 'a' ]++; } int [] map = new int [26 ]; for (int i = 0 ; i < cs.length; i++) { cnter[cs[i] - 'a' ]--; if (map[cs[i] - 'a' ] == 1 ) { continue ; } while (!stack.isEmpty() && cs[stack.getLast()] > cs[i] && cnter[cs[stack.getLast()] - 'a' ] > 0 ) { int top = stack.pollLast(); map[cs[top] - 'a' ] = 0 ; } map[cs[i] - 'a' ] = 1 ; stack.addLast(i); } StringBuilder sb = new StringBuilder (); for (int c : stack) { sb.append(cs[c]); } return sb.toString(); } }
给定 n
个非负整数,用来表示柱状图中各个柱子的高度。每个柱子彼此相邻,且宽度为 1 。
求在该柱状图中,能够勾勒出来的矩形的最大面积。
示例 1:
输入:heights = [2,1,5,6,2,3] 输出:10 解释:最大的矩形为图中红色区域,面积为 10
示例 2:
输入: heights = [2,4] 输出: 4
提示: 1 <= heights.length <=10^5 0 <= heights[i] <= 10^4
题目解析
此题属于笔试、面试都非常高频的题目,但是难度比较大。
观察可知,最终答案肯定是会完整的包含某个柱子,也就是说以某个柱子作为高,我们只需要算宽即可,而宽就是该柱子左边的第一个比他小的值和右边第一个比他小的值这个区间内。比如样例1中倒数第二个柱子:
因此解题思路如下:
统计每一个元素左边第一个比他小的元素的位置 统计每一个元素右边第一个比他小的元素的位置 遍历每一个柱子,统计最大值即可。 代码展示
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 class Solution { public int largestRectangleArea (int [] heights) { int n = heights.length; int [] rights = new int [n]; Arrays.fill(rights, n); Deque<Integer> stack = new LinkedList <>(); for (int i = 0 ; i < n; i++) { while (!stack.isEmpty() && heights[stack.getLast()] > heights[i]) { int top = stack.pollLast(); rights[top] = i; } stack.addLast(i); } int [] left = new int [n]; Arrays.fill(left, -1 ); stack.clear(); for (int i = n - 1 ; i >= 0 ; i--) { while (!stack.isEmpty() && heights[stack.getLast()] > heights[i]) { int top = stack.pollLast(); left[top] = i; } stack.addLast(i); } int res = 0 ; for (int i = 0 ; i < n; i++) { res = Math.max(res, (rights[i] - left[i] - 1 ) * heights[i]); } return res; } }
给定 n
个非负整数表示每个宽度为 1
的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水。
示例 1:
输入:height = [0,1,0,2,1,0,1,3,2,1,2,1] 输出:6 解释:上面是由数组 [0,1,0,2,1,0,1,3,2,1,2,1] 表示的高度图,在这种情况下,可以接 6 个单位的雨水(蓝色部分表示雨水)。
示例 2:
输入:height = [4,2,0,3,2,5] 输出:9
提示: n == height.length 1 <= n <= 2 * 10^4 0 <= height[i] <= 10^5
题目解析
此题也是面试笔试的高频题,也是具有比较大的难度。
如果柱子一直是单调递减的,那么是不可能接的到雨水的,只有遇到单调递减后递增才会产生凹槽,才可以接到雨水。因此我们需要用单调栈来记录每一个柱子下一个更大的柱子的位置,在这个过程中进行计算。
能够接收雨水的数量取决于“凹槽”的面积,但是由于“凹槽”的形状可能是非矩形的,所以我们需要分层计算。
比如下图:
能够接雨水的多少我们划分成如下几部分:
其中,
A的体积为: (3号柱子的高度) * (5号柱子和3号柱子之间的距离);B的体积为:(2号柱子的高度 - 3号柱子的高度) * (5号柱子和2号柱子之间的距离);C的体积为:(1号柱子的高度 - 5号柱子的高度) * (5号柱子和1号柱子之间的距离)
C的体积的运算方式是不一样的,因为并不满足height[1] < height[5];因此这里得做特殊处理。
代码展示
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 class Solution { public int trap (int [] height) { int ans = 0 ; Deque<Integer> stack = new LinkedList <>(); int n = height.length; for (int i = 0 ; i < n; i++) { int last = 0 ; while (!stack.isEmpty() && height[stack.getLast()] <= height[i]) { int top = stack.pollLast(); ans += (i - top - 1 ) * (height[top] - last); last = height[top]; } if (!stack.isEmpty()) { ans += (i - stack.getLast() - 1 ) * (height[i] - last); } stack.addLast(i); } return ans; } }