Merage Sort以及两道leetCode的相关问题

merage sort

Divide-and-conquer

merge sort 的核心理念是 Divide-and-conquer ,这个范式的核心是把问题分割成跟原问题相似的子问题,然后,递归的解决这些子问题,最后把这些子问题的结论合并得到原始问题的答案。Divide-and-conquer 分三步:

  1. Divide 把问题分割成跟原来的问题一致但是规模变小了的子问题。
  2. Conquer 递归的解决子问题。如果问题足够小了,直接解决子问题。
  3. Combine 把子问题的解决方案合并的到原问题的解决方案。

如图所示:

mergeSort1.png

进一步扩展成更多的递归步骤:

![merge_sort_recursion.png](http://upload-images.jianshu.io/upload_images/3077991-03c40937f59045ae.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)
dac.png

Merge Sort

Merge Sort 采用的就是Divide-and-conquer ,对于一个数组a[p....r]有以下三步:

  1. Divide 找到pr的中点q;
  2. Conquer 递归的对每次生成的子数组进行排序;
  3. Combine 把两个排序好的子数组合并成一个排序好的数组。

举一个具体例子:

​ 对于一个数组 array[0..7] [14, 7, 3, 12, 9, 11, 6, 2] .

  • divide 阶段,我们计算出来一个中值q=3
  • conquer 阶段,我们分别对两个子数组array[0..3], [14, 7, 3, 12]array[4..7], [9, 11, 6, 2] 进行排序。当我们完成conquer ,两个数组都是有序的分别是 [3, 7, 12, 14][2, 6, 9, 11],最后我们得到完整的数组[3, 7, 12, 14, 2, 6, 9, 11]
  • 最后在combine阶段,我们合并(merge) 两个数组,得到最终的有序数组[2, 3, 6, 7, 9, 11, 12, 14]

我们用图形来演示一下这个过程:

![Uploading Merge-sort-example-300px_933644.gif . . .]

再来两个动图演示一下:

Merge-sort-example-300px.gif
Merge_sort_animation2.gif

Java 实现

public void mergeSort(int[] A, int start, int end, int[] temp) {
    if(start >= end) {
        return;
    }
    int left = start;
    int right = end;
    int mid = (start + end) / 2;
    
    mergeSort(A, start, mid, temp);
    mergeSort(A, mid + 1, end, temp);
    merge(A, start, mid, end, temp)
}

public void merge(int[] A, int start, int mid, int end, int[] temp) {
    int left = start;
    int right = mid + 1;
    int index = start;
    
    while(left <= mid && right <= end) {
        if(A[left] < A[right]) {
            temp[index++] = A[left++];
        } else {
            temp[index++] = A[right++];
        }
    }
  while(left <= mid) {
        temp[index++] = A[left++];
  } 
  while (right <= end) {
        temp[index++] = A[right++];
  }
  for(index = start; index <= end; index++) {
        A[index] = temp[index];
  }
}

LeetCode 衍生问题

算法前面说的比较清楚,直接说两个比较难的问题:

Merge k sorted linked lists and return it as one sorted list. Analyze and describe its complexity.

public class ListNode {
    int val;
    ListNode next;
    ListNode(int x) {val = x;}
}

解法1:

ListNode dummy = new ListNode(0);

public ListNode mergeKLists(ListNode[] list) {
    if(lists.length == 0) return null;
    int i = 0;
    int j = lists.length - 1;
    while( j != 0 ) {
        while(i < j) {
            lists[i] = mergeTwo(lists[i++], lists[j--]); 
        }
        i = 0;
    }
    return lists[0];
}

public ListNode mergeTwo(ListNode node1, ListNode node2) {
    ListNode head = dummy;
    while(node1 != null && node2 != null) {
        if(node1.val < node2.val) {
            head.next = node1;
            node1 = node1.next;
        }
        head = head.next;
    }
  if(node1 != null) {
        head.next = node1;
  }
  if(node2 != null) {
        head.next = node2;
  }
  return dummy.next;
}

解法2,我们借助java的优先级队列(PriorityQueue)来解这个问题:

    public ListNode mergeKLists(ListNode[] lists) {
        PriorityQueue<ListNode> heap = new PriorityQueue<>(new Comparator<ListNode>() {
            @Override
            public int compare(ListNode o1, ListNode o2) {
                return o1.val - o2.val;
            }
        });

        for(ListNode n : lists) {
            if(n != null) {
                heap.add(n);
            }
        }

        ListNode head = heap.peek();
        while(!heap.isEmpty()) {
            ListNode min = heap.poll();
            if (min.next != null) {
                heap.add(min.next);
            }
            min.next = heap.peek();
        }
        return head;
    }

优先级队列实现了一种通过堆排序来解决问题的方案。

第二个问题 Merge Intervals:

Given a collection of intervals, merge all overlapping intervals.

For example,

Given [1,3],[2,6],[8,10],[15,18],

return [1,6],[8,10],[15,18].

数据结构如下:

 public class Interval {
        int start;
        int end;

        Interval() {start = 0; end = 0;}
        Interval(int s, int e) {start = s; end = e;}
    };

解法1:

 public List<Interval> merge(List<Interval> intervals) {
        if (intervals == null || intervals.size() <=1 ) return intervals;
        List<Interval> merged = new ArrayList<>();

        //对输入进行一下排序
        Collections.sort(intervals, new Comparator<Interval>() {
            @Override
            public int compare(Interval o1, Interval o2) {
                if(o1.start != o2.start) {
                    return o1.start - o2.start;
                }
                return o1.end - o2.end;
            }
        });

        int startMin = intervals.get(0).start;
        int endMax = intervals.get(0).end;

        for (int i = 1; i < intervals.size(); i++) {
            int start = intervals.get(i).start;
            int end = intervals.get(i).end;

            if (start > endMax) {
                merged.add(new Interval(startMin, endMax));
                //完成一个节点的添加;
                startMin = start;
                endMax = end;
            } else {
                //吞掉一个节点;
                endMax = Math.max(end, endMax);
            }
        }
        merged.add(new Interval(startMin, endMax));
        return merged;
    }

解法2(原地排序,空间占用率低):

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

推荐阅读更多精彩内容