• 插入排序(Insertion Sort)


    插入排序(Insertion Sort)

    一、基本思想

    插入排序的工作原理是通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应位置并插入。插入排序在实现上,通常采用in-place排序(即只需用到O(1)的额外空间的排序),因而在从后向前扫描过程中,需要反复把已排序元素逐步向后挪位,为最新元素提供插入空间。

    二、实现逻辑

    有一组数字:{5, 2, 4, 6, 1, 3},将这组数字从小到大进行排列。从第二个数字开始,将其认为是新增加的数字,第二个数字只需与其左边的第一个数字比较后排好序;在第三个数字,认为前两个已经排好序的数字为手里整理好的牌,那么只需将第三个数字与前两个数字比较即可;以此类推,直到最后一个数字与前面的所有数字比较结束,插入排序完成。

    算法步骤:

    1. 从第一个元素开始,该元素可以认为已经被排序;
    2. 取出下一个元素,在已经排序的序列中从后向前扫描;
    3. 如果已排序序列中的某元素大于新元素,则将该元素移到下一位置;
    4. 重复步骤 3,直到找到已排序的元素小于或等于新元素的位置;
    5. 将新元素插入到该位置;
    6. 重复步骤 2 ~ 5。

    三、时间复杂度的分析

    如果插入排序的目标是把 n n n 个元素的序列升序排序,那么:

    • 若序列已经是升序排列,只需要比较 n − 1 n-1 n1 次即可,即最好时间复杂度为: Ω ( n ) \Omega(n) Ω(n)
    • 若序列是降序排列,那么由于序列至多有 n ( n − 1 ) / 2 n(n-1)/2 n(n1)/2 对逆序,即最坏时间复杂度为: O ( n 2 ) O(n^2) O(n2)

    对于平均时间复杂度而言,长度为 n n n 的序列有 n ! n! n! 种排列,故每一种序列的排列平均有 n ( n − 1 ) / 4 n(n-1)/4 n(n1)/4 对逆序,故有平均时间复杂度为: Θ ( n 2 ) \Theta(n^2) Θ(n2)

    四、空间复杂度的分析

    由于算法采用的是in-place排序方法,不会额外开辟 O ( n ) O(n) O(n) 的数组空间(out-place排序),所以空间复杂度为: O ( 1 ) O(1) O(1)

    五、算法实现

    直接插入

    def insertion_sort(array: list, reverse: bool=False) -> None:
        '''
        array: 支持数值型数据,如整型与浮点型混合;支持全为字符串类型的数据;不支持字符串型与数值型混合。
        reverse: 是否降序, 默认采用升序。
        '''
        for index in range(1, len(array)):
            key = array[index]
            pre = index - 1
            while pre >= 0 and (key > array[pre] if reverse else key < array[pre]):
                array[pre + 1] = array[pre]
                pre -= 1
            array[pre + 1] = key
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    折半插入

    def insertion_sort(array: list, reverse: bool=False) -> None:
        '''
        array: 支持数值型数据,如整型与浮点型混合;支持全为字符串类型的数据;不支持字符串型与数值型混合。
        reverse: 是否降序, 默认采用升序。
        '''
        for index in range(1, len(array)):
            key = array[index]
            low, high = 0, index - 1
            while low <= high: # 符合单调性的序列
                mid = (low + high) // 2
                if (key < array[mid] if reverse else key > array[mid]):
                    low = mid + 1
                else:
                    high = mid - 1
            for pre in range(index, low, -1): # 从后往前
                array[pre] = array[pre - 1]
            array[low] = key
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
  • 相关阅读:
    AD637使用笔记
    一文详解Web自动化测试
    JavaScript函数
    Oracle导出sequence
    C语言实现单链表
    Spring Security - 如何修复 WebSecurityConfigurerAdapter 已弃用
    MYSQL的主从复制
    文件夹怎么隐藏?电脑重要文件夹隐藏的方法
    springmvc:设置后端响应给前端的json数据转换成String格式
    如何通过ADB命令的方式关闭华为系手机的emui系统更新升级?解决:error: no devices/emulators found
  • 原文地址:https://blog.csdn.net/linjing_zyq/article/details/126156051