C++實(shí)現(xiàn)LeetCode(84.直方圖中最大的矩形)
[LeetCode] 84. Largest Rectangle in Histogram 直方圖中最大的矩形
Given n non-negative integers representing the histogram's bar height where the width of each bar is 1, find the area of largest rectangle in the histogram.
Above is a histogram where width of each bar is 1, given height = [2,1,5,6,2,3].
The largest rectangle is shown in the shaded area, which has area = 10 unit.
For example,
Given height = [2,1,5,6,2,3],
return 10.
這道題讓求直方圖中最大的矩形,剛開始看到求極值問題以為要用DP來做,可是想不出遞推式,只得作罷。這道題如果用暴力搜索法估計(jì)肯定沒法通過OJ,有一種很好的優(yōu)化方法,就是遍歷數(shù)組,每找到一個(gè)局部峰值(只要當(dāng)前的數(shù)字大于后面的一個(gè)數(shù)字,那么當(dāng)前數(shù)字就看作一個(gè)局部峰值,跟前面的數(shù)字大小無關(guān)),然后向前遍歷所有的值,算出共同的矩形面積,每次對(duì)比保留最大值。這里再說下為啥要從局部峰值處理,看題目中的例子,局部峰值為 2,6,3,我們只需在這些局部峰值出進(jìn)行處理,為啥不用在非局部峰值處統(tǒng)計(jì)呢,這是因?yàn)榉蔷植糠逯堤幍那闆r,后面的局部峰值都可以包括,比如1和5,由于局部峰值6是高于1和5的,所有1和5能組成的矩形,到6這里都能組成,并且還可以加上6本身的一部分組成更大的矩形,那么就不用費(fèi)力氣去再統(tǒng)計(jì)一個(gè)1和5處能組成的矩形了。代碼如下:
解法一:
// Pruning optimize class Solution { public: int largestRectangleArea(vector<int> &height) { int res = 0; for (int i = 0; i < height.size(); ++i) { if (i + 1 < height.size() && height[i] <= height[i + 1]) { continue; } int minH = height[i]; for (int j = i; j >= 0; --j) { minH = min(minH, height[j]); int area = minH * (i - j + 1); res = max(res, area); } } return res; } };
后來又在網(wǎng)上發(fā)現(xiàn)一種比較流行的解法,是利用棧來解,可參考其他文檔,但是經(jīng)過仔細(xì)研究,其核心思想跟上面那種剪枝的方法有異曲同工之妙,這里維護(hù)一個(gè)棧,用來保存遞增序列,相當(dāng)于上面那種方法的找局部峰值。我們可以看到,直方圖矩形面積要最大的話,需要盡可能的使得連續(xù)的矩形多,并且最低一塊的高度要高。有點(diǎn)像木桶原理一樣,總是最低的那塊板子決定桶的裝水量。那么既然需要用單調(diào)棧來做,首先要考慮到底用遞增棧,還是用遞減棧來做。我們想啊,遞增棧是維護(hù)遞增的順序,當(dāng)遇到小于棧頂元素的數(shù)就開始處理,而遞減棧正好相反,維護(hù)遞減的順序,當(dāng)遇到大于棧頂元素的數(shù)開始處理。那么根據(jù)這道題的特點(diǎn),我們需要按從高板子到低板子的順序處理,先處理最高的板子,寬度為1,然后再處理旁邊矮一些的板子,此時(shí)長(zhǎng)度為2,因?yàn)橹暗母甙遄涌山M成矮板子的矩形 ,因此我們需要一個(gè)遞增棧,當(dāng)遇到大的數(shù)字直接進(jìn)棧,而當(dāng)遇到小于棧頂元素的數(shù)字時(shí),就要取出棧頂元素進(jìn)行處理了,那取出的順序就是從高板子到矮板子了,于是乎遇到的較小的數(shù)字只是一個(gè)觸發(fā),表示現(xiàn)在需要開始計(jì)算矩形面積了,為了使得最后一塊板子也被處理,這里用了個(gè)小 trick,在高度數(shù)組最后面加上一個(gè)0,這樣原先的最后一個(gè)板子也可以被處理了。由于棧頂元素是矩形的高度,那么關(guān)鍵就是求出來寬度,那么跟之前那道 Trapping Rain Water 一樣,單調(diào)棧中不能放高度,而是需要放坐標(biāo)。由于我們先取出棧中最高的板子,那么就可以先算出長(zhǎng)度為1的矩形面積了,然后再取下一個(gè)板子,此時(shí)根據(jù)矮板子的高度算長(zhǎng)度為2的矩形面積,以此類推,知道數(shù)字大于棧頂元素為止,再次進(jìn)棧,巧妙的一比!關(guān)于單調(diào)棧問題可以參見博主的一篇總結(jié)帖 LeetCode Monotonous Stack Summary 單調(diào)棧小結(jié),代碼如下:
解法二:
class Solution { public: int largestRectangleArea(vector<int> &height) { int res = 0; stack<int> st; height.push_back(0); for (int i = 0; i < height.size(); ++i) { if (st.empty() || height[st.top()] < height[i]) { st.push(i); } else { int cur = st.top(); st.pop(); res = max(res, height[cur] * (st.empty() ? i : (i - st.top() - 1))); --i; } } return res; } };
我們可以將上面的方法稍作修改,使其更加簡(jiǎn)潔一些:
解法三:
class Solution { public: int largestRectangleArea(vector<int>& heights) { int res = 0; stack<int> st; heights.push_back(0); for (int i = 0; i < heights.size(); ++i) { while (!st.empty() && heights[st.top()] >= heights[i]) { int cur = st.top(); st.pop(); res = max(res, heights[cur] * (st.empty() ? i : (i - st.top() - 1))); } st.push(i); } return res; } };
到此這篇關(guān)于C++實(shí)現(xiàn)LeetCode(84.直方圖中最大的矩形)的文章就介紹到這了,更多相關(guān)C++實(shí)現(xiàn)直方圖中最大的矩形內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- C++實(shí)現(xiàn)LeetCode(12.整數(shù)轉(zhuǎn)化成羅馬數(shù)字)
- C++實(shí)現(xiàn)LeetCode(55.跳躍游戲)
- C++實(shí)現(xiàn)LeetCode(45.跳躍游戲之二)
- C++實(shí)現(xiàn)LeetCode(769.可排序的最大塊數(shù))
- C++實(shí)現(xiàn)LeetCode(768.可排序的最大塊數(shù)之二)
- C++實(shí)現(xiàn)LeetCode(42.收集雨水)
- C++實(shí)現(xiàn)LeetCode(11.裝最多水的容器)
- C++實(shí)現(xiàn)LeetCode(13.羅馬數(shù)字轉(zhuǎn)化成整數(shù))
相關(guān)文章
C/C++如何實(shí)現(xiàn)循環(huán)左移,循環(huán)右移
這篇文章主要介紹了C/C++如何實(shí)現(xiàn)循環(huán)左移,循環(huán)右移,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-07-07解析C++編程中異常相關(guān)的堆棧展開和throw()異常規(guī)范
這篇文章主要介紹了C++編程中異常相關(guān)的堆棧展開和throw()異常規(guī)范,throw()規(guī)范部分文中結(jié)合了C++11標(biāo)準(zhǔn)的新特性來講,需要的朋友可以參考下2016-01-01C語言 function recursion函數(shù)遞歸詳解
遞歸指的是在函數(shù)的定義中使用函數(shù)自身的方法,舉個(gè)例子: 從前有座山,山里有座廟,廟里有個(gè)老和尚,正在給小和尚講故事呢!故事是什么呢?"從前有座山,山里有座廟,廟里有個(gè)老和尚,正在給小和尚講故事呢!故事是什么呢?"從前有座山,山里有座廟,循環(huán)下去2021-10-10VC中使用ADO開發(fā)數(shù)據(jù)庫應(yīng)用程序簡(jiǎn)明教程
這篇文章主要介紹了VC中使用ADO開發(fā)數(shù)據(jù)庫應(yīng)用程序的方法,結(jié)合實(shí)例形式詳細(xì)講述了ADO的原理及VC使用ADO開發(fā)數(shù)據(jù)庫應(yīng)用程序的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2016-06-06C語言實(shí)現(xiàn)簡(jiǎn)單的定時(shí)器
這篇文章主要為大家詳細(xì)介紹了C語言實(shí)現(xiàn)簡(jiǎn)單的定時(shí)器,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-10-10C++ 漢諾塔問題知識(shí)點(diǎn)總結(jié)
在本篇文章里小編給大家整理的是關(guān)于C++ 漢諾塔問題知識(shí)點(diǎn)內(nèi)容,有需要的朋友們可以參考下。2020-02-02