Leetcode - Largest Rectangle in Histogram

Question:

Paste_Image.png

My code:

import java.util.Stack;

public class Solution {
    public int largestRectangleArea(int[] height) {
        if (height == null || height.length == 0)
            return 0;
        int count = height.length;
        int[] area = new int[count + 1];
        for (int i = 0; i < count; i ++)
            area[i] = height[i];
        area[count] = 0;
        int maxArea = 0;
        Stack<Integer> s = new Stack<Integer>();
        for (int i = 0; i < count + 1; i++) {
            if (s.isEmpty() || area[i] >= area[s.peek()])
                s.push(i);
            else {
                while (!s.isEmpty() && area[i] < area[s.peek()]) {
                    int top = area[s.pop()];
                    int currArea = top * (s.isEmpty() ? i : i - s.peek() - 1);
                    if (currArea > maxArea)
                        maxArea = currArea;
                }
                s.push(i);
            }
        }
        return maxArea;
    }
    public static void main(String[] args) {
        int[] height = {2, 1, 5, 6, 2, 3};
        Solution test = new Solution();
        System.out.println(test.largestRectangleArea(height));
    }
}

My test result:

Paste_Image.png

这次作业是因为之前的 Max Square才找到的。因为之前看了相关的思路,所以这道题目就比较顺得写下来了。虽然最后还是出了一些小问题。
他的思路是什么呢?
其实就是将数由小到大放进栈中。当碰到一个比栈顶数(当前最大数)小的数时,将栈中比该数大的数都pop出来,算面积。然后将该数压入栈中。也就说,栈中的数一直是从小到大排列的。然后碰到小的,就将比其大的pop出来,一个个算面积,然后求出最大面积。
因为值大的不会和值小的构成矩形,只会和值大的构成矩形。所以比该数大的就全部pop出来。然后计算出来的面积是完整的。然后比该值小的数依旧存在栈里面,以后该数pop时,需要连同这些小的一起计算面积。
核心原则是:
小的不能和大的构成矩形算面积。
大的可以和小的构成矩形算面积(切掉头上方那些多余部分)。

**
总结:还是一种思路问题吧。我觉得用栈这种数据结构用的很妙。同样需要考虑的问题是,什么样的数可以进入栈。pop出栈的数据又需要满足哪些条件。
**

有点疲倦。但写着写着脑子又清醒了。

Anyway, Good luck, Richardo!

My code:

public class Solution {
    public int largestRectangleArea(int[] height) {
        if (height == null || height.length == 0)       
            return 0;
        Stack<Integer> s = new Stack<Integer>();
        int[] helper = new int[height.length + 1];
        for (int i = 0; i < height.length; i++)
            helper[i] = height[i];
        int max = 0;
        int zeroLocation = -1;
        for (int i = 0; i < helper.length; i++) {
            if (s.isEmpty() || helper[i] >= helper[s.peek()]) { // insert into stack with increasing elem values
                s.push(i);
            }
            else {
                int end = i; // get the end index
                while (!s.isEmpty()) {
                    if (helper[s.peek()] >= helper[i]) { // if bigger, pop, area = value * (end - begin)
                        int index = s.pop();
                        int area = 0;
                        if (s.isEmpty()) {
                            int begin = zeroLocation;
                            area = helper[index] * (i - begin - 1);
                        }
                        else {
                            int begin = s.peek();
                            area = helper[index] * (i - begin - 1);
                        }
                        max = Math.max(max, area); // get the biggest area
                    }
                    else
                        break;
                }
                if (helper[i] > 0)
                    s.push(i); // push this index into stack
                else
                    zeroLocation = i;
            }
        }
        return max;
    }
}

自己又写了一遍,终于写出来了。。花费了大约一小时,许多尝试。
最大的问题出在:
左边的边界是 s.peek() + 1;
所以我还考虑了0的情况,觉得0不该插入栈中,然后拿一个变量来记录每一个0的位置,然后算出每段的区域面积,如果栈空了,就 i - zeroLocation + 1
这个挺麻烦的。然后用我以前写的代码,网上看的,就简洁很多。0也插入进去。
如果栈空了,之前的数中一定没存在过0,否则0不会弹出。
然后,算总面积,宽为 i
如果栈不空,就拿 s.peek() + 1作为左边界,不用再去更新zeroLocation,对所有数字也是统一处理,不需要分非0或0.
简洁了很多。
这道题目是做

  1. Maximal Rectangle
    http://www.jianshu.com/p/c86e5dec7514
    想起来的。之前也做过这道题目,所以知道思路,就是把他当做等高地图那样来做。每个值表示的就是一个直方图,然后将每行的直方图作为参数送到这个函数里求出最大面积。
    遍历所有行,直到找到最大值。

Anyway, Good luck, Richardo!

My code:

public class Solution {
    public int largestRectangleArea(int[] heights) {
        if (heights == null || heights.length == 0) {
            return 0;
        }
        Stack<Integer> s = new Stack<Integer>();
        int[] h = new int[heights.length + 1];
        for (int i = 0; i < heights.length; i++) {
            h[i] = heights[i];
        }
        int max = 0;
        for (int i = 0; i < h.length; i++) {
            if (s.isEmpty() || h[i] >= h[s.peek()]) {
                s.push(i);
            }
            else {
                while (!s.isEmpty() && h[i] < h[s.peek()]) {
                    int high = h[s.pop()];
                    int end = i;
                    int begin = (s.isEmpty() ? 0 : s.peek() + 1);
                    int area = high * (end - begin);
                    max = Math.max(max, area);
                }
                s.push(i);
            }
        }
        return max;
    }
}

差不多的思路。一个栈。
最重要的问题是,
弹出一个值算面积,一定要首先确定他的begin and end

Anyway, Good luck, Richardo! -- 08/18/2016

My code:

public class Solution {
    public int largestRectangleArea(int[] heights) {
        if (heights == null || heights.length == 0) {
            return 0;
        }
        int[] nums = new int[heights.length + 1];
        for (int i = 0; i < heights.length; i++) {
            nums[i] = heights[i];
        }
        Stack<Integer> st = new Stack<Integer>();
        int i = 0;
        int max = 0;
        while (i < nums.length) {
            if (st.isEmpty() || nums[st.peek()] <= nums[i]) {
                st.push(i);
                i++;
            }
            else {
                int index = st.pop();
                int right = i - 1;
                int left = st.isEmpty() ? 0 : st.peek() + 1;
                int area = (right - left + 1) * nums[index];
                max = Math.max(max, area);
            }
        }
        
        return max;
    }
}

注意找到这个直方块,比他大的最左边和最右边界限。

Anyway, Good luck, Richardo! -- 09/17/2016

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

推荐阅读更多精彩内容