数据结构与算法----插入排序与Shell排序(Python版)

一、插入排序Insertion Sort

插入排序时间复杂度仍然是O(n2),但算法思路与冒泡排序、选择排序不同

插入排序维持一个已排好序的子列表,其位置始终在列表的前部,然后逐步扩大这个子列表直到全表

1趟,子列表仅包含第1个数据项,将第2个数据项作为新项插入到子列表的合适位置中,这样已排序的子列表就包含2个数据项

2趟,再继续将第3个数据项跟前2个数据项比对,并移动比自身大的数据项,空出位置来,以便加入到子列表中

经过n-1趟比对和插入,子列表扩展到全表,排序完成

插入排序的比对主要用来寻找新项

插入位置

最差情况是每趟都与子列表中所有项进行比对,总比对次数与冒泡排序相同,数量级仍是O(n2)

最好情况,列表已经排好序的时候,每趟仅需1次比对,总次数是O(n)

插入排序:思路

源码:

def insertSort(alist):
    for index in range(len(alist)):
        #插入项
        currentvalue = alist[index]
        position = index

        while position > 0 and alist[position-1] > currentvalue:
            alist[position] = alist[position-1]  #移动对比
            position = position - 1

        alist[position] = currentvalue
    return alist

testlist = [1,2,6,4,8,5,7,3]
print(insertSort(testlist))

 

由于移动操作仅包含1次赋值,是交换操作 的1/3,所以插入排序性能会较好一些
 

二、谢尔排序Shell Sort

我们注意到插入排序的比对次数,在最好的情况下是O(n),这种情况发生在列表已是有序的情况下,实际上,列表越接近有序,插入排序的比对次数就越少

从这个情况入手,谢尔排序以插入排序作为基础,对无序表进行间隔划分子列表,每个子列表都执行插入排序

随着子列表的数量越来越少,无序表的整 体越来越接近有序,从而减少整体排序的 比对次数
间隔为 3 的子列表,子列表分别插入排序 后的整体状况更接近有序
谢尔排序:思路
最后一趟是标准的插入排序,但由于前面 几趟已经将列表处理到接近有序,这一趟 仅需少数几次移动即可完成
子列表的间隔一般从 n/2 开始,每趟倍增 n/4, n/8 …… 直到 1

 

def shellSort(alist):
    sublistcount = len(alist)//2  #间隔设定
    while sublistcount > 0 :
        for startposition in range(sublistcount):
            #调用插入排序为子系列排序
            alist = gapInsertSort(alist,startposition,sublistcount)
        sublistcount = sublistcount//2  #缩小间隔
    return alist

def gapInsertSort(alist,start,gap):
    for index in range(start+gap,len(alist),gap):
        currentvalue = alist[index]
        position = index
        while position >=gap and alist[position-gap] > currentvalue:
            alist[position] = alist[position - gap]
            position = position - gap

        alist[position] = currentvalue

    return alist

testlist = [1,2,6,4,8,5,7,3]
print(shellSort(testlist))
源码:

谢尔排序:算法分析

粗看上去,谢尔排序以插入排序为基础,可能并不会比插入排序好

但由于每趟都使得列表更加接近有序,这过程会减少很多原先需要的无效比对

对谢尔排序的详尽分析比较复杂,大致说是介于O(n)O(n2)之间

如果将间隔保持在2k-1(13571531等等),谢尔排序的时间复杂度约为On3/2)

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 
 
 
Python

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
十大经典排序算法是指在计算机科学中常用的排序算法,它们分别是: 1. 冒泡排序(Bubble Sort):重复地比较相邻的两个元素,将较大的元素逐渐向右移动。 2. 选择排序(Selection Sort):每次从未排序的部分选择最小(或最大)的元素,并放在已排序的部分的末尾。 3. 插入排序(Insertion Sort):将未排序的元素逐个插入到已排序的部分中的正确位置。 4. 希尔排序(Shell Sort):将待排序的数组按照一定步长进行分组,对每组进行插入排序,逐渐减小步长。 5. 归并排序(Merge Sort):将待排序的数组递归地分成两半,对每一半进行排序,然后合并两个有序数组。 6. 快速排序(Quick Sort):选择一个基准元素,将数组划分为两部分,左边部分都小于基准,右边部分都大于基准,递归地对两部分进行排序。 7. 堆排序(Heap Sort):将待排序的数组构建成一个最大堆(或最小堆),然后依次取出堆顶元素并调整堆结构。 8. 计数排序(Counting Sort):统计数组中每个元素出现的次数,然后根据统计结果对元素进行排序。 9. 桶排序(Bucket Sort):将待排序的数组划分为多个桶,对每个桶中的元素进行排序,最后将桶中的元素按顺序合并。 10. 基数排序(Radix Sort):按照元素的位数,将待排序的数组从低位到高位进行排序。 以上是十大经典排序算法,每种算法都有其适用的场景和性能特点,选择合适的排序算法可以提高程序的效率。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值