Given a 2D binary matrix filled with 0's and 1's, find the largest rectangle containing all ones and return its area.

PS:将其化解为柱状图求最大体积的问题。先用动态规划的思路,将矩形中垂直方向的连续1的数量进行计算。

而后逐行扫描,并在扫描时将其转化为柱状图求体积。

即若高度一直增加,则持续压入栈,当出现h[i]<栈顶高度时,则出栈,并计算出栈元素的所在矩形的体积。持续计算直到栈顶元素高度小于h[i],则将h[i]压栈。

注意:计算时需要在原始数据队尾加入0以使得扫描继续到最后一个元素。此时队列长度为n+1,最大宽度n。

计算矩形时,宽度为i-s.top-1,若栈内无元素,则表明h[i]之前的所有元素的高度都<h[i],则宽度为i。

首先我们看一下下面的例子:

height的内容是 [5,6,7,8,3],特点是除了最后一个,前面全部保持递增,且最后一个立柱的高度小于前面所有立柱高度。

对于这种特点的柱状图,如果使用上面所说的“挨个使用每一个柱状图的高度作为矩形的高度,求面积”的方法,还需要用嵌套循环吗?

我们知道除了最后一个,从第一个到倒数第二个立柱的高度都在升高,那么如果挨个使用每一个柱的高度作为矩形的高度,那么依次能得到的矩形的宽度就可以直接算出来:使用5作为高度可以使用前四个立柱组成 4*5的矩形,高度6可以组成3*6的矩形... 因此只需要遍历一次,选出最大面积即可。

对于这种类型的柱状图,最大矩形面积的时间复杂度是O(n)。

我们将这种特点的柱状图称为“波峰图”。

下面介绍新的解法的步骤:

(1) 在height尾部添加一个0,也就是一个高度为0的立柱。作用是在最后也能凑成上面提的那种“波峰图”。

(2) 定义了一个stack,然后遍历时如果height[i] 大于stack.top(),进栈。反之,出栈直到栈顶元素小于height[i]。

由于出栈的这些元素高度都是递增的,我们可以求出这些立柱中所围成的最大矩形。更妙的是,由于这些被弹出的立柱处于“波峰”之上(比如弹出i 到 i+k,那么所有这些立柱的高度都高于 i-1和 i+k+1的高度),因此,如果我们使用之前所提的“左右延伸找立柱”的思路解,以这些立柱的高度作为整个矩形的高度时,左右延伸出的矩形所包含的立柱不会超出这段“波峰”,因为波峰外的立柱高度都比他们低。“波峰图”其实就是求解最大矩形的“孤岛”,它不会干扰到外部。

(3) 由于比height[i]大的元素都出完了,height[i]又比栈顶元素大了,因此再次进栈。如此往复,直到遍历到最后那个高度为0的柱,触发最后的弹出以及最后一次面积的计算,此后stack为空。

(4) 返回面积最大值。

栈中存的不是高度,而是height的索引,这样做的好处是不会影响宽度的计算,索引值相减 = 宽度。

 class Solution {
public:
int maximalRectangle(vector<vector<char> > &matrix) {
int m=matrix.size();
int n=matrix[].size();
if(m<||n<) return ;
vector<vector<int>> v(m,vector<int>(n,));
for(int i=;i<n;i++)
v[][i]=matrix[][i]=='';
for(int i=;i<n;i++){
for(int j=;j<m;j++){
v[j][i]=matrix[j][i]==''?v[j-][i]+:;
}
}
int max=;
for(int i=;i<m;i++){
int tmp=sol(v[i]);
if(tmp>max) max=tmp;
}
return max;
}
int sol(vector<int> v){
v.push_back();
int n=v.size();
stack<int> s;
int res=;
for(int i=;i<n;i++){
while(!s.empty()&&v[s.top()]>=v[i]){
int index=s.top();
s.pop();
int max=v[index]*(s.empty()?i:(i-s.top()-));
if(max>res) res=max;
}
s.push(i);
}
return res;
}
};

最新文章

  1. Spring AOP支持的AspectJ切入点语法大全
  2. Qt动画效果展示(文艺IT男)
  3. Cordova - 与iOS原生代码交互1(通过JS调用Swift方法)
  4. 20145213《Java程序设计》第十周学习总结
  5. centos将自编译安装的apache添加为linux系统服务
  6. WCF 服务调用RFC 出现异常
  7. Fedora 17 修改GRUB启动菜单顺序
  8. Oracle 行拼接 wmsys.wm_concat扩展
  9. 微信小程序 picker 中range-key的坑
  10. MSCKF_VIO:MSCKF的双目版本
  11. 转载:c++深拷贝和浅拷贝
  12. 使用正则真正的修改TP5的config.php文件
  13. MySQL:ROWNUM的假实现
  14. BZOJ1179或洛谷3672 [APIO2009]抢掠计划
  15. HTML字体的设置
  16. Educational Codeforces Round 44 (Rated for Div. 2)
  17. PHP中str_replace和substr_replace有什么区别?
  18. 黄聪:V2010中C#实现友好的等待任务完成时,出现的多线程悬浮窗体
  19. dip,px,sp区别及使用场景
  20. html学习笔记--标签大全

热门文章

  1. Mysql实战之主从复制的读写分离
  2. web储存用户信息
  3. 【09】Vue 之 Vuex 数据通信
  4. FOJ Problem 2273 Triangles
  5. Linux下常用的命令记录
  6. struts2 package 属性说明
  7. webservice测试工具
  8. duilib入门简明教程 -- 界面布局(9) (转)
  9. Chrome扩展修改页面代码执行环境的方法
  10. DOM和jquery对象之间的转换