线性时间排序(计数、基数、桶排序)

线性时间排序

1.计数排序

原理:
计数排序是一种稳定的排序算法,但适用范围相对来说具有一定的局限性。当数据之间的差值太大,消耗的计算机资源相对来说太大,则不适合于用计数排序。在适合的数据面前,其排序的时间复杂度的是非常小的,运行时间为Θ(n)。

基本思想:
对于每一个输入元素x,确定小于x的元素个数。以此来确定x的输出位置,例如:小于x的元素有15个,则第16号输出元素就是x。当有几个元素相同时,将方案略作修改,因为不能将其放在同一输出位置上。

代码:

#计数排序代码
def CountingSort(a, b, k):
  c=[]
  for i in range(k+1):
    c.append(0)
  for j in range(len(a)):
    c[a[j]] = c[a[j]] + 1
  for i in range(1, k+1):
    c[i] = c[i] + c[i-1]
  for j in range(len(a)-1, -1, -1):
    b[c[a[j]]-1] = a[j]#!!!!!减一是关键
    c[a[j]] = c[a[j]] - 1
  return b
if __name__ == '__main__':
  a=[10, 17, 50, 7, 30, 24, 27, 45, 15, 5, 36, 21] 
  print(CountingSort(a, b, max(a)))

2.基数排序
原理:
基数排序就是利用桶的概念按照位数进行排序
按照个位数上面的值,将元素放入不同的桶中,等数组全部放入完毕,再将所有元素按照桶的顺序依次取出,再按照十位数上面的值进行桶分布,直到最大位数分布完毕,则第一个桶就是排完序的数组。

代码:

#基数排序代码
def radix_sort(s):
    i = 0 # 记录当前正在排拿一位,最低位为1
    max_num = max(s)  # 最大值
    j = len(str(max_num))  # 记录最大值的位数
    while i < j:
        bucket_list =[[] for _ in range(10)] #初始化桶数组
        for x in s:
            bucket_list[int(x / (10**i)) % 10].append(x) # 找到位置放入桶数组
        print(bucket_list)
        s.clear()
        for x in bucket_list:   # 放回原序列
            for y in x:
                s.append(y)
        i += 1

if __name__ == '__main__':
    a = [10, 17, 50, 7, 30, 24, 27, 45, 15, 5, 36, 21] 
    radix_sort(a)
    print(a)

3.桶排序
原理:
工作的原理是将数组分到有限数量的桶子里。每个桶子再个别排序(有可能再使用别的排序算法或是以递归方式继续使用桶排序进行排序)。
假定有1-100个编号的桶(也就是定义一个长度为100的整型一维数组),每输入一个数字·就在对应的桶上插一个小旗(也就是对应下标的桶加1次),如果这个数字出现了n次就在对应桶上插n个小旗,当所有数输入完毕时,只需要从下标1开始找那些数字是1,如果是1就打印1次,是2就打印2次,是多少就打印多少次。

代码:

#桶排序代码
def bucket_sort(the_list):
    #设置全为0的数组
    all_list = [0 for i in range(100)]
    last_list = []
    for v in the_list:
        all_list[v] = 1 if all_list[v]==0 else all_list[v]+1
    for i,t_v in enumerate(all_list):
        if t_v != 0:
            for j in range(t_v):
                last_list.append(i)
    return last_list
if __name__ == '__main__':
    the_list = [10, 17, 50, 7, 30, 24, 27, 45, 15, 5, 36, 21] 
    print ("排序前:" + str(the_list))
    print ("排序后:" + str(bucket_sort(the_list)))
  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值