为什么80%的码农都做不了架构师?>>>   

前面的三种排序算法(冒泡排序,选择排序,插入排序)在平均情况下均为O(n^2)复杂度,在处理较大数据的时候比较吃力。现在来说说相对快速一些的算法,例如下面的归并排序。

算法概述/思路

归并排序是基于一种被称为“分治”(divide and conquer)的策略。其基本思路是这样的:

1.对于两个有序的数组,要将其合并为一个有序数组,我们可以很容易地写出如下代码:

//both a and b is ascend.
public void merge(int[] a, int[] b, int[] c){ int i=0,j=0,k=0; while (i<=a.length && j<=b.length){ if (a[i]<=b[i]){ c[k++]=a[i++]; } else{ c[k++]=b[j++]; } } while (i<=a.length){ c[k++]=a[i++];  } while (j<=b.length){ c[k++]=b[j++]; }
}

容易看出,这样的合并算法是高效的,其时间复杂度可达到O(n)。

2.假如有一个无序数组需要排序,但它的两个完全划分的子数组A和B分别有序,借助上述代码,我们也可以很容易实现;

3.那么,如果A,B无序,怎么办呢?可以把它们再分成更小的数组。

4.如此一直划分到最小,每个子数组都只有一个元素,则可以视为有序数组。

5.从这些最小的数组开始,逆着上面的步骤合并回去,整个数组就排好了。

总而言之,归并排序就是使用递归,先分解数组为子数组,再合并数组。

下面是归并排序的示意图(图片来自维基百科):

代码实现

//归并排序 public static void mergeSort(int[] arr){ int[] temp =new int[arr.length]; internalMergeSort(arr, temp, 0, arr.length-1); } private static void internalMergeSort(int[] a, int[] b, int left, int right){ //当left==right的时,已经不需要再划分了 if (left<right){ int middle = (left+right)/2; internalMergeSort(a, b, left, middle);          //左子数组 internalMergeSort(a, b, middle+1, right);       //右子数组 mergeSortedArray(a, b, left, middle, right);    //合并两个子数组 } } // 合并两个有序子序列 arr[left, ..., middle] 和 arr[middle+1, ..., right]。temp是辅助数组。 private static void mergeSortedArray(int arr[], int temp[], int left, int middle, int right){ int i=left;       int j=middle+1; int k=0; while ( i<=middle && j<=right){ if (arr[i] <=arr[j]){ temp[k++] = arr[i++]; } else{ temp[k++] = arr[j++]; } } while (i <=middle){ temp[k++] = arr[i++]; } while ( j<=right){ temp[k++] = arr[j++]; } //把数据复制回原数组 for (i=0; i<k; ++i){ arr[left+i] = temp[i]; } }

需要说明的是,在合并数组的时候需要一个temp数组。我们当然有足够的理由在每次调用的时候重新new一个数组(例如,减少一个参数),但是,注意到多次的创建数组对象会造成额外的开销,我们可以在开始就创建一个足够大的数组(等于原数组长度就行),以后都使用这个数组。实际上,上面的代码就是这么写的。

算法性能/复杂度

归并排序的效率是很高的,由于递归划分为子序列只需要logN复杂度,而合并每两个子序列需要大约2n次赋值,为O(n)复杂度,因此,只需要简单相乘即可得到归并排序的时间复杂度 O(㏒n)。并且由于归并算法是固定的,不受输入数据影响,所以它在最好、最坏、平均情况下表现几乎相同,均为O(㏒n)。

但是,归并排序最大的缺陷在于其空间复杂度。从上面的代码可以看到,在合并子数组的时候需要一个辅助数组,然后再把这个数据拷贝回原数组。所以,归并排序的空间复杂度(额外空间)为O(n)。可不可以省略这个数组呢?不行!如果取消辅助数组而又要保证原来的数组中数据不被覆盖,那就必须要在数组中花费大量时间来移动数据。不仅容易出错,还降低了效率。因此这个辅助空间是少不掉的。

算法稳定性

因为我们在遇到相等的数据的时候必然是按顺序“抄写”到辅助数组上的,所以,归并排序同样是稳定算法。

算法适用场景

归并排序在数据量比较大的时候也有较为出色的表现(效率上),但是,其空间复杂度O(n)使得在数据量特别大的时候(例如,1千万数据)几乎不可接受。而且,考虑到有的机器内存本身就比较小,因此,采用归并排序一定要注意。

转载于:https://my.oschina.net/lifj/blog/389709

常见排序算法:归并排序相关推荐

  1. 常见排序算法之归并排序——归并排序

    哈喽大家好,我是保护小周ღ,本期为大家带来的是常见排序算法中的归并排序,博主在这里先分享归并排序的递归算法,包您一看就会,快来试试吧~ ​ 目录 一.归并排序 1.1 基本思想 1.2 算法思想 1. ...

  2. android studio插入数据表中没有_学Java能拿高薪吗 Java中常见排序算法有哪些

    学Java能拿高薪吗?Java中常见排序算法有哪些?作为老牌编程语言,Java拥有广阔的市场占有率,几乎90%以上的大中型互联网应用系统在服务端开发都会首选Java.为了加入到Java这一高薪行业,很 ...

  3. python实现常见排序算法

    python实现常见排序算法 快速排序 思想:取出第一个元素把它放到序列的中间某一个正确位置,以它进行分割成左边和右边,再分别对左边和右边进行取元素分割(递归) 递归实现 def quicksort( ...

  4. php 面试靠快速排序,搞定PHP面试 - 常见排序算法及PHP实现

    常见排序算法及PHP实现全文代码使用PHP7.2语法编写 0. 五种基础排序算法对比 1. 冒泡排序(Bubble Sort) 冒泡排序 是一种交换排序,它的基本思想是:对待排序记录从后往前(逆序)进 ...

  5. python常见排序算法解析

    python--常见排序算法解析 算法是程序员的灵魂. 下面的博文是我整理的感觉还不错的算法实现 原理的理解是最重要的,我会常回来看看,并坚持每天刷leetcode 本篇主要实现九(八)大排序算法,分 ...

  6. 常见排序算法的原理与实现(js)

    常见排序算法通常需要通过比较来确定次序(比较类排序):由于其时间复杂度不能突破O(nlogn),因此也称为非线性时间比较类排序. 各类排序算法的特点: 术语解释: 时间复杂度:对排序数据的总的操作次数 ...

  7. 十种常见排序算法欢聚一堂

    文章目录 1.常见排序算法分类 2.非线性时间比较类排序 2.1 交换类排序 2.1.1 冒泡排序 2.1.2 快速排序 2.2 插入类排序 2.2.1 直接插入排序 2.2.2 Shell 排序 2 ...

  8. ++代码实现 感知机的原理_常见排序算法原理及JS代码实现

    来源:SegmentFault 思否社区 作者:Warren 本文只是将作者学习的过程以及算法理解进行简单的分享,提供多一个角度的理解说明,或许让你的困惑能得以解决(代码或说明若有问题,欢迎留言.联系 ...

  9. 常见排序算法及其对应的时间复杂度、空间复杂度

    常见排序算法及其对应的时间复杂度.空间复杂度: 排序算法经过长时间演变,大体可以分为两类:内排序和外排序.在排序过程中,全部记录存放在内存,则成为内排序:如果排序过程中需要使用外存,则称为外排序,本文 ...

  10. 堆排序稳定性举例_常见排序算法的稳定性分析

    一.不稳定排序算法有哪些 1.堆排序 2.希尔排序 3.快速排序 4.选择排序 口诀:一堆(堆)希尔(希尔)快(快速)选(选择) 二.常见排序算法稳定性分析 1.堆排序稳定性分析 我们知道堆的结构是节 ...

最新文章

  1. Centos6 升级glibc-2.17,解决Requires: libc.so.6(GLIBC_2.14)(64bit)错误解决方法
  2. VTK:相互作用之ImageRegion
  3. springboot jar包部署_Spring Boot项目基于Jar部署和打包详解教程
  4. 淘宝分布式NOSQL框架:Tair
  5. 在职场,进行管理的基本思维顺序
  6. 中国顶级程序员有多牛?一个弄哭韩国,一个堪称阿里“扫地僧”
  7. DynamipsGUI 模拟pix防火墙
  8. 用R语言进行分位数回归
  9. 黑马程序员--统计字符串中大小写字母和数字的个数问题的代码
  10. unity 3D飞机打石头小游戏
  11. CFS任务的负载均衡
  12. 淘宝/天猫关键词搜索采集接口分析商品价格走势(商品列表,商品销量,商品价格,分类ID采集精准商品数据)接口代码对接流程
  13. JSON Viewer 安装
  14. Postman 批量执行请求
  15. java 虚拟机(2)
  16. 【图像加密】正交拉丁方置乱+混沌图像加密解密【含GUI Matlab源码 636期】
  17. 谷歌和哈佛团队用深度学习预测余震;谷歌移交 Kubernetes 管理权,并承诺资助 900 万美元...
  18. Revit二次开发——元素信息绑定的两种方法
  19. 日本电视台选择 Infortrend存储管理核心播放业务
  20. python证书微软认证费用_微软认证有哪些?费用多少?

热门文章

  1. 已在页面完全加载前强制排版_公众号写作排版指南v2.0(适配Dark Mode)
  2. dataguard如何实现切换_ORACLE dataguard 切换
  3. java类中的static块_java类中static代码块的执行次数
  4. oracle使用小技巧,Oracle使用技巧
  5. mysql 局域网_MySQL 局域网授权问题
  6. springboot+自定义注解实现灵活的切面配置
  7. wxpython播放视频_opencv视屏流嵌入wxpython框架
  8. ztree 标准得json数据格式_Django+zTree构建组织架构树
  9. r语言热图对列不进行聚类_R语言:手把手教你画pheatmap热图
  10. 非期望产出的sbm模型_线性模型 vs. Logistic模型——离散选择模型之二