排序算法(三)——插入排序及改进
来源:互联网 发布:js代码在线翻译 编辑:程序博客网 时间:2024/05/21 02:48
插入排序
基本思想
在要排序的一组数中,假设前面(n-1)[n>=2] 个数已经是排好顺序的,现在要把第n个数找到相应位置并插入,使得这n个数也是排好顺序的。如此反复循环,直到全部排好顺序。
java实现
算法分析
在第一趟排序中,插入排序最多比较一次,第二趟最多比较两次,依次类推,最后一趟最多比较N-1次。因此有:
1+2+3+...+N-1 =N*N(N-1)/2
因为在每趟排序发现插入点之前,平均来说,只有全体数据项的一半进行比较,我们除以2得到:
N*N(N-1)/4
复制的次数大致等于比较的次数,然而,一次复制与一次比较的时间消耗不同,所以相对于随机数据,这个算法比冒泡排序快一倍,比选择排序略快。
与冒泡排序、选择排序一样,插入排序的时间复杂度仍然为O(N2),这三者被称为简单排序或者基本排序,三者都是稳定的排序算法。
如果待排序数组基本有序时,插入排序的效率会更高。
插入排序的改进
在插入某个元素之前需要先确定该元素在有序数组中的位置,上例的做法是对有序数组中的元素逐个扫描,当数据量比较大的时候,这是一个很耗时间的过程,可以采用二分查找法改进,这种排序也被称为二分插入排序。
改进后的代码如下:
还有一种在二分插入排序的基础上进一步改进的排序,称为2-路插入排序,其目的是减少排序过程中移动记录的次数,但为此需要n个记录的辅助空间。
算法的思想为:另设一个和原始待排序列L相同的数组D,首先将L[1]复制给D[1],并把D[1]看成是已排好序的序列中处于中间位置的元素(枢纽元素),之后将L中的从第二个元素开始依次插入到数组D中,大于D[1]的插入到D[1]之后的序列(此处我称为右半边序列,用的是数组左半部分空间),小于D[1]的插入到D[1]之前的序列(左半边序列,用的是数组右半部分空间)。
该算法将数组当做首尾衔接的环形结构来使用。
示意图如下:
排序完成之后,数组中的元素并不是按照下标升序排列的,而是靠first与final指针确定起始元素。
注意:当L[1]为最小值时,2-路插入排序失去它的优越性,等同于二分插入排序。
代码如下:
如果对如下数组进行排序
8,1,11,12,4,20,7,2,6,15
打印结果如下:
此时,first指向下标为5的元素(1),last指向下标为4的元素(20)
- 排序算法(三)——插入排序及改进
- 排序算法(三)——插入排序及改进
- 排序算法(三)——插入排序及改进
- 排序算法(一)——冒泡排序及改进
- 排序算法(二)——选择排序及改进
- 排序算法(一)——冒泡排序及改进
- 排序算法(二)——选择排序及改进
- 排序算法(一)——冒泡排序及改进
- 排序算法(二)——选择排序及改进
- 排序算法(1) —— 直接插入排序及其改进
- 排序算法(1) —— 直接插入排序及其改进
- 排序算法三——插入排序
- 排序(二)插入排序、插入改进——二分插入排序、插入改进——希尔排序
- 排序算法自我练习(三)——插入排序
- 排序算法(三)——插入排序
- 排序算法(三) —— 插入排序
- 插入排序及插入排序改进
- 改进的插入排序算法
- f-stack dpdk bond 功能
- git学习指南
- Echarts多张图片插入word解决方案
- 如何优雅地使用Sublime Text3
- A+B problem(0)
- 排序算法(三)——插入排序及改进
- 白天谋生存,晚上谋发展
- bionic
- Android studio导入含有jni代码依赖工程的eclipse项目及常见错误解决
- ubuntu 主机无法解析问题
- (2) JAVA:线性表的顺序存储与链式存储
- 如何使用Python自带编辑器IDLE
- Windows搭建Sosoapi
- 排序算法(四)——归并排序与递归