Leetcode - Insert Interval

Screenshot from 2016-01-11 18:59:19.png

My code:

/**
 * Definition for an interval.
 * public class Interval {
 *     int start;
 *     int end;
 *     Interval() { start = 0; end = 0; }
 *     Interval(int s, int e) { start = s; end = e; }
 * }
 */
public class Solution {
    public List<Interval> insert(List<Interval> intervals, Interval newInterval) {
        ArrayList<Interval> ret = new ArrayList<Interval>();
        if (intervals == null || intervals.size() == 0) {
            ret.add(newInterval);
            return ret;
        }
        if (newInterval == null)
            return intervals;
        ret.add(newInterval);
        for (int i = 0; i < intervals.size(); i++) {
            Interval originInterval = intervals.get(i);
            int originStart = originInterval.start;
            int originEnd = originInterval.end;
            for (int j = ret.size() - 1; j >= -1; j--) {
                if (j == -1) { // if it is minimun, insert at the beginning whose index is 0
                    ret.add(0, originInterval);
                    break;   
                }
                Interval retInterval = ret.get(j);
                int retStart = retInterval.start;
                int retEnd = retInterval.end;
                if (originStart > retEnd) { // if bigger than this element, insert behind it, so the inserting index should be j + 1
                    ret.add(j + 1, originInterval);
                    break;
                }
                else if (originEnd < retStart) { // if smaller than this element, visit the front elements to compare
                    continue;
                }
                else {
                    retInterval.start = Math.min(originStart, retStart); // if needs merge, change values so to merge
                    retInterval.end = Math.max(originEnd, retEnd);
                    break;
                }
            }
        }
        return ret;
    }
}

这道题木之前做过,但应该没有写文章。但还有印象。
自己写了出来。
其实就是一个插入 + merge的过程。
怎么样才能让这个过程足够简单并且可以用程序实现。
This is the question we need to deal with.
Fuck off!
如果就在 intervals上实现这么一个过程。
那么会很复杂,思考起来。
你想,
你如何判断什么时候,该merge?
那么,我现在的思路是,
假设:
originStart, originEnd 为intervals其中某个元素的起始。
newStart, newEnd 为需要插入元素的起始。
So,

for (int i = 0; i < intervals.size(); i++) {
     if (newStart > originEnd)
           continue;
     else if (newEnd < originStart)
           intervals.add(i, newInterval);
     else {
           
     }
}

想着想着,突然发现也可以做,而且更加简洁,只要一次遍历就行。。。
于是写了出来。
我他妈真他妈,变强了,草。
**
但是奇怪的一点,
我觉得这么做速度更快,但是LJ上显示,速度反而更慢了,怎么会呢?不能理解。到时候得问下人。
**

我一开始的做法就是新建一个集合,然后先把newInterval放进去,然后把intervals里面的元素一个个有序放进去,需要merge的地方就merge,然后就写出来了。
第二个做法就是针对intervals来做。
思路在上面的伪代码上基本上也写明白了。
但是,为什么速度反而慢了很多呢?

/**
 * Definition for an interval.
 * public class Interval {
 *     int start;
 *     int end;
 *     Interval() { start = 0; end = 0; }
 *     Interval(int s, int e) { start = s; end = e; }
 * }
 */
public class Solution {
    public List<Interval> insert(List<Interval> intervals, Interval newInterval) {
        ArrayList<Interval> ret = new ArrayList<Interval>();
        if (intervals == null || intervals.size() == 0) {
            ret.add(newInterval);
            return ret;
        }
        if (newInterval == null)
            return intervals;
        ret.add(newInterval);
        int newStart = newInterval.start;
        int newEnd = newInterval.end;
        int i = 0;
        for (; i < intervals.size(); i++) {
            int originStart = intervals.get(i).start;
            int originEnd = intervals.get(i).end;
            if (newStart > originEnd)
                continue;
            else if (newEnd < originStart) {
                intervals.add(i, new Interval(newStart, newEnd));
                break;
            }
            else { // update newStart, newEnd,remove origin element, later will add updated element(merge)
                newStart = Math.min(originStart, newStart);
                newEnd = Math.max(originEnd, newEnd);
                intervals.remove(i);
                i--;
            }
        }
        if (i == intervals.size())
            intervals.add(new Interval(newStart, newEnd));
        return intervals;
    }
}

这道题目级别是hard,但是现在想来没有什么难度。
DP才是王道啊。

Anyway, Good luck, Richardo!

More efficient solution:
My code:

/**
 * Definition for an interval.
 * public class Interval {
 *     int start;
 *     int end;
 *     Interval() { start = 0; end = 0; }
 *     Interval(int s, int e) { start = s; end = e; }
 * }
 */
public class Solution {
    public List<Interval> insert(List<Interval> intervals, Interval newInterval) {
        ArrayList<Interval> ret = new ArrayList<Interval>();
        if (intervals == null || intervals.size() == 0) {
            ret.add(newInterval);
            return ret;
        }
        for (int i = 0; i < intervals.size(); i++) {
            Interval curr = intervals.get(i);
            if (curr.end < newInterval.start) {
                ret.add(curr);
            }
            else if (curr.start > newInterval.end) {
                ret.add(newInterval);
                newInterval = curr;
            }
            else if (curr.start <= newInterval.start || curr.end >= newInterval.end) {
                newInterval = new Interval(Math.min(curr.start, newInterval.start), Math.max(curr.end, newInterval.end));
            }
        }
        ret.add(newInterval);
        return ret;
    }
}

Anyway, Good luck, Richardo!

My code:

/**
 * Definition for an interval.
 * public class Interval {
 *     int start;
 *     int end;
 *     Interval() { start = 0; end = 0; }
 *     Interval(int s, int e) { start = s; end = e; }
 * }
 */
public class Solution {
    public List<Interval> insert(List<Interval> intervals, Interval newInterval) {
        List<Interval> ret = new ArrayList<Interval>();
        if (intervals == null || intervals.size() == 0) {
            ret.add(newInterval);
            return ret;
        }
        else if (newInterval == null) {
            return intervals;
        }
        
        Interval curr = newInterval;
        for (int i = intervals.size() - 1; i >= 0; i--) {
            Interval pre = intervals.get(i);
            if (pre.end < curr.start) {
                intervals.add(i + 1, curr);
                return intervals;
            }
            else if (pre.start <= curr.start) {
                pre.end = Math.max(pre.end, curr.end);
                return intervals;
            }
            else if (pre.start <= curr.end) {
                pre.start = curr.start;
                pre.end = Math.max(pre.end, curr.end);
                curr = pre;
                intervals.remove(i);
            }
            else {
                continue;
            }
        }
        
        intervals.add(0, curr);
        return intervals;
    }
}

和merge interval差不多,是他的一个子问题。

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

My code:

/**
 * Definition for an interval.
 * public class Interval {
 *     int start;
 *     int end;
 *     Interval() { start = 0; end = 0; }
 *     Interval(int s, int e) { start = s; end = e; }
 * }
 */
public class Solution {
    public List<Interval> insert(List<Interval> intervals, Interval newInterval) {
        List<Interval> ret = new ArrayList<Interval>();
        int i = 0;
        for (; i < intervals.size(); i++) {
            Interval curr = intervals.get(i);
            if (curr.end < newInterval.start) {
                ret.add(curr);
                continue;
            }
            else {
                break;
            }
        }
        
        for (; i < intervals.size(); i++) {
            Interval curr = intervals.get(i);
            if (curr.start > newInterval.end) {
                break;
            }
            newInterval.start = Math.min(newInterval.start, curr.start);
            newInterval.end = Math.max(newInterval.end, curr.end);
        }
        
        ret.add(newInterval);
        for (; i < intervals.size(); i++) {
            ret.add(intervals.get(i));
        }
        
        return ret;
    }
}

reference:
https://discuss.leetcode.com/topic/7808/short-and-straight-forward-java-solution

这个做法比之前的做法更高效,可以保证 O(n) 的复杂度

Anyway, Good luck, Richardo! -- 10/15/2016

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

推荐阅读更多精彩内容