题目链接/文章讲解:代码随想录
思考:本题和接雨水类似,42. 接雨水是找每个柱子左右两边第一个大于该柱子高度的柱子,而本题是找每个柱子左右两边第一个小于该柱子的柱子。
栈顶和栈顶的下一个元素以及要入栈的三个元素组成了我们要求最大面积的高度和宽度
三种情况:
- 情况一:当前遍历的元素heights[i]大于栈顶元素heights[st.top()]的情况
- 情况二:当前遍历的元素heights[i]等于栈顶元素heights[st.top()]的情况
- 情况三:当前遍历的元素heights[i]小于栈顶元素heights[st.top()]的情况
另外本题需要在height数组的前后都加一个0,因为
1.如果数组本身是升序的
例如[2,4,6,8],那么入栈之后都是单调递减,一直都没有走 情况三 计算结果的哪一步,所以最后输出的就是0了。
结尾加一个0,就会让栈里的所有元素,走到情况三的逻辑。
2.如果数组本身是降序的
例如 [8,6,4,2],在 8 入栈后,6 开始与8 进行比较,此时我们得到 mid(8),rigt(6),但是得不到 left,因此需要加入一个0
代码实现:
- // 版本一
- class Solution {
- public:
- int largestRectangleArea(vector<int>& heights) {
- int result = 0;
- stack<int> st;
- heights.insert(heights.begin(), 0); // 数组头部加入元素0
- heights.push_back(0); // 数组尾部加入元素0
- st.push(0);
-
- // 第一个元素已经入栈,从下标1开始
- for (int i = 1; i < heights.size(); i++) {
- if (heights[i] > heights[st.top()]) { // 情况一
- st.push(i);
- } else if (heights[i] == heights[st.top()]) { // 情况二
- st.pop(); // 这个可以加,可以不加,效果一样,思路不同
- st.push(i);
- } else { // 情况三
- while (!st.empty() && heights[i] < heights[st.top()]) { // 注意是while
- int mid = st.top();
- st.pop();
- if (!st.empty()) {
- int left = st.top();
- int right = i;
- int w = right - left - 1;
- int h = heights[mid];
- result = max(result, w * h);
- }
- }
- st.push(i);
- }
- }
- return result;
- }
- };
精简:
- // 版本二
- class Solution {
- public:
- int largestRectangleArea(vector<int>& heights) {
- stack<int> st;
- heights.insert(heights.begin(), 0); // 数组头部加入元素0
- heights.push_back(0); // 数组尾部加入元素0
- st.push(0);
- int result = 0;
- for (int i = 1; i < heights.size(); i++) {
- while (heights[i] < heights[st.top()]) {
- int mid = st.top();
- st.pop();
- int w = i - st.top() - 1;
- int h = heights[mid];
- result = max(result, w * h);
- }
- st.push(i);
- }
- return result;
- }
- };