深入浅出排序算法之直接插入排序(拓展:折半插入排序)

简介: 深入浅出排序算法之直接插入排序(拓展:折半插入排序)

直接插入排序和选择排序的第一趟就是第一个关键字

1. 图示解析

2. 原理解析

整个区间被分为:① 有序区间;② 无序区间

每次选择无序区间的第一个元素,在有序区间内选择合适的位置插入。

为了各位小伙伴能更加清楚地认识直接插入排序,我接下用文字描述直接插入排序的过程!

想通过一个例子来体会一下插入排序的执行流程。例如,对原始序列{49,38,65,97,76,13,27,49}进行直接插入排序的具体流程如下(序列中有两个49,其中一个加下划线,加以区分)。

原始序列:49   38   65   97   76   13   27   49

(1)一开始只看49,一个数当然是有序的。

有序序列:{49};无序序列:{38   65   97   76   13   27   49}

(2)向有序序列插入38,38 < 49,所以49向后移动一个位置,38插入到49原来的位置,这趟排序后的结果为:

有序序列:{38   49};无序序列:{ 65   97   76   13   27   49}

(3)向有序序列插入65,65  > 49,所以不需要移动,65就应该在49只有,这趟排序后的结果为:

有序序列:{38   49   65};无序序列:{97   76   13   27   49}

(4)插入97,97 > 65,所以不需要移动,97就应该在65之后,这趟排序后的结果为:

有序序列:{38   49   65   97};无序序列:{76   13   27   49}

(5)插入76,76 < 97,所以97向后移动一个位置,继续比较,76 > 65,65不需要移动,76应该插入在65之后,97之前,这趟排序后的结果为:

有序序列:{38   49   65   76   97};无序序列:{13   27   49}

(6)插入13,13 < 97,97后移;13 < 76,76后移;这样逐个向前比较,发现13应该插入在最前面,这趟排序后的结果为:

有序序列:{13   38   49   65   76   97};无序序列:{27   49}

(7)插入27,还是从后面前行比较,确定27应该插入在13之后、38之前,这趟排序后的结果为:

有序序列:{13   27   38   49   65   76   97};无序序列:{49}

(8)最后插入49,同样从后向前逐个比较,知道49 = 49 < 65,它的位置确定,直接插入排序全过程完成。最后的排序结果为:

有序序列:{13   27   38   49   49   65   76   97};无序序列:{}

总结算法思想:

每趟将一个待排序的关键字按照其值的大小插入到已经拍好的部分有序序列的位置上直到所有待排序关键字都插入到有序序列中为止。

注意!!!!!!!!!!

小伙伴们请注意,我这里省略了i = 0的情况,从i = 1开始,也就是一开始只看49,一个数当然是有序的。如果是在考试中一定要把i = 0的情况作为第一趟(针对专升本和考研都一样)。

3. 代码实现

//直接插入排序
    public static void insertSort(int[] arr) {
        //代码可以从i = 1开始算起,但是做题画图时,一定要从i = 0开始算起
        for (int i = 1; i < arr.length; i++) {
            int j = i - 1;
            int tmp = arr[i];
            for (; j >= 0; j--) {
                //如果arr[j] > tmp变成arr[j] >= tmp就变成不稳定了
                if (arr[j] > tmp) {
                    arr[j + 1] = arr[j];
                } else {
                    break;
                }
            }
            arr[j + 1] = tmp;
        }
    }
    public static void main(String[] args) {
        int[] arr = {10,9,8,7,6,5,4,3,2,1};
        Sort.insertSort(arr);
        for (int x : arr) {
            System.out.print(x + " ");
        }
    }

4. 性能分析

时间复杂度 空间复杂度
最好 平均 最坏
O(N) O(N^2) O(N^2) O(1)
数据有序 数据逆序

稳定性:稳定

如果一个排序是稳定的,那么就可以实现为不稳定的

但是如果一个算法本身是不稳定的,你没办法实现为稳定的排序

插入排序,原始数据越有序,时间效率越高!

检测一下:

//创建升序数组
    public static void createArray1(int[] arr){
        for(int i = 1;i<10000;i++){
            arr[i - 1] = i;
        }
    }
    //创建逆序数组
    public static void createArray2(int[] arr){
        for(int i = 0;i<10000;i++){
            arr[i] = 10000 - i;
        }
    }
    public static void main(String[] args) {
        int[] arr1 = new int[10000];
        Test.createArray1(arr1);
        long s1 = System.currentTimeMillis();
        Sort.insertSort(arr1);
        long e1 = System.currentTimeMillis();
        System.out.println("数组有序的情况:"+(e1 - s1));
        int[] arr2 = new int[10000];
        Test.createArray2(arr2);
        long s2 = System.currentTimeMillis();
        Sort.insertSort(arr2);
        long e2 = System.currentTimeMillis();
        System.out.println("数组逆序的情况:"+(e2 - s2));
    }

 

5. 折半插入排序(拓展)

在有序区间选择数据应该插入的位置时,因为区间的有序性,可以利用折半查找的思想来增加插入算法的效率!

//折半插入排序
    public static void bsInsertSort(int[] arr) {
        //代码可以从i = 1开始算起,但是做题画图时,一定要从i = 0开始算起
        for (int i = 1; i < arr.length; i++) {
            int v = arr[i];
            int left = 0;//左边标记永远从0下标开始
            int right = i;
            while(left < right){
                int mid = (left + right) / 2;
                //需要[left right)
                if(arr[mid] < v){
                    //如果区间要闭,就赋值mid + 1或者mid - 1
                    left = mid + 1;
                }else{
                    //如果右区间要开,就赋值mid
                    right = mid;
                }
            }
            for (int j = i; j > left; j--) {
                arr[j] = arr[j - 1];
            }
            arr[left] = v;
        }
    }
相关文章
|
3月前
|
算法 搜索推荐
数据结构与算法学习十一:冒泡排序、选择排序、插入排序
本文介绍了冒泡排序、选择排序和插入排序三种基础排序算法的原理、实现代码和测试结果。
28 0
数据结构与算法学习十一:冒泡排序、选择排序、插入排序
|
3月前
|
算法 Java 索引
数据结构与算法学习十五:常用查找算法介绍,线性排序、二分查找(折半查找)算法、差值查找算法、斐波那契(黄金分割法)查找算法
四种常用的查找算法:顺序查找、二分查找(折半查找)、插值查找和斐波那契查找,并提供了Java语言的实现代码和测试结果。
42 0
|
3月前
|
搜索推荐 算法
【排序算法(一)】——插入排序,选择排序 —> 深层解析
【排序算法(一)】——插入排序,选择排序 —> 深层解析
|
6月前
|
算法 搜索推荐 C#
|
7月前
|
机器学习/深度学习 算法 搜索推荐
数据结构算法--2 冒泡排序,选择排序,插入排序
**基础排序算法包括冒泡排序、选择排序和插入排序。冒泡排序通过相邻元素比较交换,逐步将最大值“冒”到末尾,平均时间复杂度为O(n^2)。选择排序每次找到剩余部分的最小值与未排序部分的第一个元素交换,同样具有O(n^2)的时间复杂度。插入排序则类似玩牌,将新元素插入到已排序部分的正确位置,也是O(n^2)复杂度。这些算法适用于小规模或部分有序的数据。**
|
7月前
|
算法 搜索推荐
数据结构与算法-插入排序
数据结构与算法-插入排序
40 2
|
7月前
|
算法 搜索推荐 数据可视化
【漫画算法】插入排序:插入宝石的传说
【漫画算法】插入排序:插入宝石的传说
|
7月前
|
人工智能 搜索推荐 JavaScript
心得经验总结:排序算法:插入排序法(直接插入法和希尔排序法)
心得经验总结:排序算法:插入排序法(直接插入法和希尔排序法)
55 0
|
7月前
|
机器学习/深度学习 搜索推荐 算法
【C/排序算法】:直接插入排序和希尔排序
【C/排序算法】:直接插入排序和希尔排序
54 0
|
7月前
|
搜索推荐 算法
排序算法之插入排序
排序算法之插入排序
61 0