# 84 柱形图中最大的矩形
想了半天,好难,不会做。本来想用dp做,递推公式想不出来。但是请记住,递推公式发现太麻烦复杂想不出来,也是一种进展!说明我们发现用dp表不合适,要换一种方法了!
以下讲了两种方法,但不管哪种,都是要找到每个柱子,左边第一个小于自己的,和右边第一个小于自己的,如下图。
方法一:双指针法
其实我觉得不是严格意义的双指针?
不懂为什么随想录说是为了防止死循环。
大概思路是图里这样,对每个柱子,相当于找高度和他高度一样的矩形
while (t >= 0 && heights[t] >= heights[i]) t = minLeftIndex[t];
minLeftIndex[i] = t;
这里我觉得好难理解。其实就是现在要找i左边第一个小于他的,左边第一个是t,不满足。接下来我们要找的还是要比i小的,那至少也要比t小,既然t的左边第一个小于的已经找到了,那我们就用他的。但也要重新和t代表的值比一下大小。
int largestRectangleArea(vector<int>& heights) {
vector<int> minLeftIndex(heights.size());
vector<int> minRightIndex(heights.size());
int size = heights.size();
// 每柱子 左第一个小于该柱子的下标
minLeftIndex[0] = -1; // 注意这里初始化,防止下面while死循环
for (int i = 1; i < size; i++) {
int t = i - 1;
while (t >= 0 && heights[t] >= heights[i]) t = minLeftIndex[t];
minLeftIndex[i] = t;
}
// 每柱子 右第一个小于该柱子的下标
minRightIndex[size - 1] = size; // 注意这里初始化,防止下面while死循环
for (int i = size - 2; i >= 0; i--) {
int t = i + 1;
while (t < size && heights[t] >= heights[i]) t = minRightIndex[t];
minRightIndex[i] = t;
}
int result = 0;
for (int i = 0; i < size; i++) {
int sum = heights[i] * (minRightIndex[i] - minLeftIndex[i] - 1);
result = max(sum, result);
}
return result;
}
方法二:单调栈
"本题是要找每个柱子左右两边第一个小于该柱子的柱子,所以从栈头(元素从栈头弹出)到栈底的顺序应该是从大到小的顺序!"
大部分逻辑和接雨水一样。但有个关键:要在vec的头尾各加上一个0元素。
末尾加0的原因:如果数组升序例如[2,4,6,8],那么入栈之后 都是单调递减,一直都没有走 情况三(新来的<top) 计算结果的步,所以最后输出的就是0了
开头加0的原因:如果数组降序如8642,那8入栈之后,6发现比8小,然后mid是8,right是6那个index,但是把8 pop以后stack已经空了,我们找不到left。反正也进不去stack不为空的那个if,就计算不了了。所以为了让第一个柱子为高度的长方形也计算,就要开头加个0.
接雨水不用开头结尾加0的原因:接雨水找到高的之后,是算里面形成的凹槽。但是长方形是要把找到的柱子也包括进去的,就会有问题。而且接雨水如果是单调递减,就存不住雨水,确实答案要为0.
int largestRectangleArea(vector<int>& vec) {
int res=0;
stack<int> st;
vec.push_back(0);
vec.insert(vec.begin(),0);
st.push(0);
for(int i=1;i<vec.size();i++){
if(vec[i]>=vec[st.top()]) st.push(i);
else{
while(!st.empty() && vec[i]<vec[st.top()]){
int h=vec[st.top()];
st.pop();
if(!st.empty()){
int right=i;
int left=st.top();
res=max(res,h*(right-left-1));
}
}
st.push(i);
}
}
return res;
}
# 两个月训练营总结
--今天正好也把已经更新的图论做完了(额外题目没做)。2023-7-22 一刷结束撒花!二刷开始!
--养成了非常好的写博客的习惯,我会把我犯了错误的点,关键思路,难想通的逻辑,新学到的知识和写法都写上去。确实有点耗时间,但现在回过头去觉得自己是扎扎实实写过来的。希望二刷的时候因为自己一刷这么认真写了笔记,不要忘的太快。
--卡哥的内容真的非常好,讲的很清晰透彻,非常感谢;训练营也提供了一个很好的氛围,觉得是在和大家一起努力。大部分题目我觉得我是完全弄懂了(记没记住,下次还会不会做就不知道了),我没弄懂的地方都在博客里标注了橙色,二刷再来看。
--做题速度和熟练程度比刚开始好太多了。但现在还是由于我有点固执死脑筋+认真写博客导致 一个题(自己做/尝试+学习随想录+写博客)至少要花30min,长的要花1h。 还是希望自己能灵活一点,效率高一点。我还是有点高中那种,知识点没完全想通就不放过的脑子(完全是和没学会就背下来相反的那种),但现在应试要紧我觉得还是不要这么死脑筋!
真的请自己:不要死磕不要死磕不要死磕浪费时间
--其实训练营大部分时间没跟上进度,猛补了一段时间在50天左右终于跟上了进度。我的工作模式是,有时候彻底摆烂一题不做,有时候一天做十几道题,有点极端了。还是希望自己能养成持续努力的习惯,每天坚持中高强度,让自己和时间一起发挥作用。