数据结构与算法之直接插入排序及其改进(可视化)

1. 插入排序

1.1 简介

  插入排序的基本思路是每一次遍历的时候将当前元素插入到之前已经有序的子数组中。
  插入排序伪代码:

function insert_sort(list):
    for i from 1 to list.length - 1:
        j = i
        while list[i - 1] > list[i] && j > 0:
            swap(list[i - 1], list[i]) 

1.2 python代码实现

def insert_sort(l, start, end):
    '''
    @brief  插入排序,排序范围[start, end]
    @param  l   进行排序的list
    @param  start   开始位置
    @param  end 结束位置
    '''
    count = 0
    for i in range(start, end + 1):
        j = min(end, i + 1)
        while j > start and l[j - 1] > l[j]:
            l[j], l[j - 1] = l[j - 1], l[j]
            j -= 1

1.3 流程

在这里插入图片描述

1.4 可视化

在这里插入图片描述

2. 插入排序改进II

2.1 思路

  直接插入排序需要不断的交换两个元素可以引入一个哨兵,每次交换只写一次数组提升性能。具体操作是在每次遍历前记录下当前位置的元素值,遍历结束后放置到目标位置。

2.2 python代码实现

def insert_sort_II(l, start, end):
    '''
    @brief  插入排序,排序范围[start, end],排序时使用哨兵,减少对数组的访问
    @param  l   进行排序的list
    @param  start   开始位置
    @param  end 结束位置
    '''
    for i in range(start + 1, end + 1):
        tmp = l[i]
        j = i - 1
        while j >= start and l[j] > tmp:
            l[j + 1] = l[j]
            j -= 1
            
        l[j + 1] = tmp

3 插入排序改进III

3.1 思路

  插入排序是向已经有序的子数组中插入当前值,既然是向有序数组中插入完全可以使用二分查找法查找目标插入区域,然后将目标值交换到具体位置。

3.2 python 代码实现

def insert_sort_III(l, start, end):
    '''
    @brief  插入排序,排序范围[start, end],排序时使用哨兵,减少对数组的访问
    @param  l   进行排序的list
    @param  start   开始位置
    @param  end 结束位置
    @note   使用二分查找法优化    
    '''
    for i in range(start + 1, end + 1):
        j = i
        left = start
        right = i
        while left <= right:
            mid = int(left + (right - left)/2)
            if l[mid] < l[j]:
                left = mid + 1
            else:
                right = mid - 1
        
        j = i
        value = l[j]
        while j > left:
            l[j] = l[j - 1]
            j -= 1
            
        l[j] = value

4. 性能对比

  注:下面的图是用我的笔记本测试的不是很稳定,之后有时间用服务器测试下。

数据分布性能对比
无序在这里插入图片描述
有序在这里插入图片描述
逆序在这里插入图片描述
90%有序在这里插入图片描述
高斯分布在这里插入图片描述
泊松分布在这里插入图片描述
所有元素相同在这里插入图片描述
数组中只有两个值在这里插入图片描述

5. 复杂度分析

算法时间复杂度空间复杂度最坏比较次数最佳比较次数
直接插入排序算法I O ( n 2 ) O(n^2) O(n2) O ( 1 ) O(1) O(1) n ( n − 1 ) 2 \frac{n(n-1)}{2} 2n(n1) n − 1 n-1 n1
插入排序算法II O ( n 2 ) O(n^2) O(n2) O ( 1 ) O(1) O(1) n ( n − 1 ) 2 \frac{n(n-1)}{2} 2n(n1) n − 1 n-1 n1
插入排序算法III O ( n 2 ) O(n^2) O(n2) O ( 1 ) O(1) O(1) n l o g n nlog{n} nlogn n − 1 n-1 n1
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值