排序算法5:归并排序(Merge sort)

今天药忘吃喽~ 2021-09-28 22:18 512阅读 0赞

归并排序(Merge sort)

归并排序(Merge sort)是建立在归并操作上的一种有效的排序算法。该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。

作为一种典型的分而治之思想的算法应用,归并排序的实现由两种方法:

自上而下的递归(所有递归的方法都可以用迭代重写,所以就有了第 2 种方法);
自下而上的迭代;
在《数据结构与算法 JavaScript 描述》中,作者给出了自下而上的迭代方法。但是对于递归法,作者却认为:

However, it is not possible to do so in JavaScript, as the recursion goes too deep for the language to handle.

然而,在 JavaScript 中这种方式不太可行,因为这个算法的递归深度对它来讲太深了。

说实话,我不太理解这句话。意思是 JavaScript 编译器内存太小,递归太深容易造成内存溢出吗?还望有大神能够指教。

和选择排序一样,归并排序的性能不受输入数据的影响,但表现比选择排序好的多,因为始终都是 O(nlogn) 的时间复杂度。代价是需要额外的内存空间。

2. 算法步骤

申请空间,使其大小为两个已经排序序列之和,该空间用来存放合并后的序列;

设定两个指针,最初位置分别为两个已经排序序列的起始位置;

比较两个指针所指向的元素,选择相对小的元素放入到合并空间,并移动指针到下一位置;

重复步骤 3 直到某一指针达到序列尾;

将另一序列剩下的所有元素直接复制到合并序列尾。

3. 动图演示

在这里插入图片描述

java实现

  1. public class MergeSort implements IArraySort {
  2. @Override
  3. public int[] sort(int[] sourceArray) throws Exception {
  4. // 对 arr 进行拷贝,不改变参数内容
  5. int[] arr = Arrays.copyOf(sourceArray, sourceArray.length);
  6. if (arr.length < 2) {
  7. return arr;
  8. }
  9. int middle = (int) Math.floor(arr.length / 2);
  10. int[] left = Arrays.copyOfRange(arr, 0, middle);
  11. int[] right = Arrays.copyOfRange(arr, middle, arr.length);
  12. return merge(sort(left), sort(right));
  13. }
  14. protected int[] merge(int[] left, int[] right) {
  15. int[] result = new int[left.length + right.length];
  16. int i = 0;
  17. while (left.length > 0 && right.length > 0) {
  18. if (left[0] <= right[0]) {
  19. result[i++] = left[0];
  20. left = Arrays.copyOfRange(left, 1, left.length);
  21. } else {
  22. result[i++] = right[0];
  23. right = Arrays.copyOfRange(right, 1, right.length);
  24. }
  25. }
  26. while (left.length > 0) {
  27. result[i++] = left[0];
  28. left = Arrays.copyOfRange(left, 1, left.length);
  29. }
  30. while (right.length > 0) {
  31. result[i++] = right[0];
  32. right = Arrays.copyOfRange(right, 1, right.length);
  33. }
  34. return result;
  35. }
  36. }

分析过程

在这里插入图片描述
可以看到这种结构很像一棵完全二叉树,本文的归并排序我们采用递归去实现(也可采用迭代的方式去实现)。分阶段可以理解为就是递归拆分子序列的过程,递归深度为log2n。

合并相邻有序子序列

再来看看治阶段,我们需要将两个已经有序的子序列合并成一个有序序列,比如上图中的最后一次合并,要将[4,5,7,8]和[1,2,3,6]两个已经有序的子序列,合并为最终序列[1,2,3,4,5,6,7,8],来看下实现步骤。

在这里插入图片描述

发表评论

表情:
评论列表 (有 0 条评论,512人围观)

还没有评论,来说两句吧...

相关阅读

    相关 归并排序merge sort

    归并排序(Merge sort)是建立在归并操作上的一种有效的排序算法。该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。(出自[维基百科][Li

    相关 算法5--排序--Merge Sorted Array

    之前几天在忙其他的事情,没有时间更新,今天更新了几个,虽然有几个SMR的博客暂时没有开放,已经写好了,以后会慢慢开放的 今天再更新一个有关排序的算法题 1 Merge So