Leetcode 85. Maximal Rectangle

Problem

Given a 2D binary matrix filled with 0's and 1's, find the largest rectangle containing only 1's and return its area.
For example, given the following matrix:
1 0 1 0 0
1 0 1 1 1
1 1 1 1 1
1 0 0 1 0
return 6

Think

这个问题对我而言还是有一点难度的,主要参考了youtube上的一个视频:https://www.youtube.com/watch?v=g8bSdXCG-lA&t=71s
这个问题解答可以分解为两个步骤,在这之前需要考虑一个叫做
max Area Rectangle in histogram的问题(直方图中最大的矩形面积)
这个问题的描述用一个图还是一目了然的,不需要太多解释.

image.png

现在我们假设解决了max Area Rectangle in histogram,现在考虑怎么解决本来需要解决的问题.
  现在有输入数据:

1 0 1 0 0
1 0 1 1 1
1 1 1 1 1
1 0 0 1 0

  • 用一个dp数组保存"直方图"数据,数组的大小与行同长(len(dp) == 5) dp[i]可以理解为以第i 行为x轴绘制的直方图纵轴高度数据.
  • 依照行循环
     第1次循环 dp[0] = [1,0 ,1, 0, 0 ]
    第2次: dp[1] = [2, 0, 2, 1, 1]
     第3次: dp[2] = [3, 1, 2, 2, 2]
     第4次: dp[3] = [4, 0, 0, 3, 0]
  • 依次计算上述四次循环中dp数据的max Area Rectangle in histogram值,四次循环中的最大值就是最终答案

max Area Rectangle in histogram 问题

这个问题其实也并不简单,特别是如果要用O(N)复杂度解决问题的话,还是比较考验智商的.
关于这个问题的解决方案可以参考:
http://www.geeksforgeeks.org/largest-rectangle-under-histogram/
这个是一个非常巧妙的解决方案,基本思想是使用stack

// The main function to find the maximum rectangular area under given
// histogram with n bars
int getMaxArea(int hist[], int n)
{
    // Create an empty stack. The stack holds indexes of hist[] array
    // The bars stored in stack are always in increasing order of their
    // heights.
    stack<int> s;
 
    int max_area = 0; // Initalize max area
    int tp;  // To store top of stack
    int area_with_top; // To store area with top bar    as the smallest bar
 
    // Run through all bars of given histogram
    int i = 0;
    while (i < n)
    {
        // If this bar is higher than the bar on top stack, push it to stack
        if (s.empty() || hist[s.top()] <= hist[i])
            s.push(i++);
        // If this bar is lower than top of stack, then calculate area of rectangle 
        // with stack top as the smallest (or minimum height) bar. 'i' is 
        // 'right index' for the top and element before top in stack is 'left index'
        else
        {
            tp = s.top();  // store the top index
            s.pop();  // pop the top
            cout<<"the top is:"<<hist[tp]<<endl;
            // Calculate the area with hist[tp] stack as smallest bar
            area_with_top = hist[tp] * (s.empty() ? i : i - s.top() - 1);
            cout<<"max area:"<<area_with_top<<endl;
            // update max area, if needed
            if (max_area < area_with_top)
                max_area = area_with_top;
        }
    }
 
    // Now pop the remaining bars from stack and calculate area with every
    // popped bar as the smallest bar
    while (s.empty() == false)
    {
        tp = s.top();
        s.pop();
        cout<<"top is:"<<hist[tp]<<endl;
        area_with_top = hist[tp] * (s.empty() ? i : i - s.top() - 1);
        cout<<"area:"<<area_with_top<<endl;
        if (max_area < area_with_top)
            max_area = area_with_top;
    }
 
    return max_area;
}
int main(){
int hist[] = {4,2,5,3,1,6};
    int n = sizeof(hist)/sizeof(hist[0]);
    cout << "Maximum area is " << getMaxArea(hist, n)<<endl;
    return 0;
}

参考资源:https://www.youtube.com/watch?v=KkJrGxuQtYo
这里所使用的原理是: 对于每一个高度,计算在这个高度上能够达到的最大面积值,然后在这些最大面积值中进行比较,得出最终结果.在某一个高度上能够带到的最大面积由"边界"决定,所谓边界指的是两边的高度都比指定高度要低.

Code

class Solution(object):
    
    def maxAreaRectangle(self, seq):
        # type(seq): list(int)
        # list[i] represent height at i
        max_area = -1
        i = 0
        stack = []
        while i < len(seq):
            if len(stack) == 0 or seq[stack[-1]] <= seq[i]:
                stack.append(i)
                i += 1
            else:
                tp = stack.pop()
                area_with_top = seq[tp] * (i if len(stack) == 0 else (i - stack[-1] - 1))
                if area_with_top > max_area:
                    max_area = area_with_top
        while len(stack) > 0:
            tp = stack.pop()
            area_with_top = seq[tp] * (i if len(stack) == 0 else (i - stack[-1] - 1))
            if area_with_top > max_area:
                max_area = area_with_top
        return max_area

    def maximalRectangle(self, matrix):
        """
        :type matrix: List[List[str]]
        :rtype: int
        """
        m = len(matrix)
        if m == 0:
            return 0
        n = len(matrix[0])
        ans = -1
        dp = []
        for i in range(m):
            if i == 0:
                for j in range(n):
                    dp.append(int(matrix[i][j]))
            else:
                new_dp = []
                for j in range(n):
                    if dp[j] != 0 and matrix[i][j] == '1':
                        new_dp.append(dp[j] + 1)
                    else:
                        new_dp.append(int(matrix[i][j]))
                dp = new_dp
            level_max = self.maxAreaRectangle(dp)
            if level_max > ans:
                ans = level_max
        return ans
```  
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,980评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,178评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,868评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,498评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,492评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,521评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,910评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,569评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,793评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,559评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,639评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,342评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,931评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,904评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,144评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,833评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,350评论 2 342

推荐阅读更多精彩内容