代码随想录算法训练营第60天 | 单调栈 ●84 柱形图中最大的矩形 ●两个月训练营总结

# 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天左右终于跟上了进度。我的工作模式是,有时候彻底摆烂一题不做,有时候一天做十几道题,有点极端了。还是希望自己能养成持续努力的习惯,每天坚持中高强度,让自己和时间一起发挥作用。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值