网站首页 > 技术文章 正文
排序算法
排序算法简单地分为两类:比较排序和非比较排序,其中比较排序是通过比较元素的相对大小来实现排序,其复杂度的上限为O(nlogn),所以也称其为非线性时间排序;非比较排序不是通过比较元素相对大小来实现,通常能够以O(n)的复杂度来实现,所以也称其为线性时间排序,下面给出了十种排序算法的复杂度(时间复杂度和空间复杂度)和稳定性
其中,n 表示需要排序的元素数量,k 表示桶的数量
LB三人组
冒泡排序
冒泡排序的思想就是每次循环过程中,大的元素下降,小的元素上升,从而进行排序,具体过程如下:
代码如下:
def bubble_sort(a):
n = len(a)
flag = False #设置标志位,避免无效排序
for i in range(n-1, 0, -1):
for j in range(i):
if a[j] > a[j + 1]:
a[j], a[j + 1] = a[j + 1], a[j]
flag = True
if not flag:
break
return a
if __name__ == "__main__":
a = [90, 5, 83, 42, 12, 15]
print(bubble_sort(a))
选择排序
选择排序的思想就是每次循环的过程中找到最小的值放在第一位,依次循环找到其他较小的值放在接下来的位置,通过查找进行排序,其过程如下:
代码如下:
def select_sort(a):
n = len(a)
for i in range(n - 1):
min_index = i
for j in range(i+1, n):
if a[j] < a[min_index]:
min_index = j
a[i], a[min_index] = a[min_index], a[i]
return a
if __name__ == "__main__":
a = [90, 5, 83, 42, 12, 15]
print(select_sort(a))
插入排序
插入排序的思想就是和之前的元素进行比较,大的元素放在后面,小的元素放在前面,具体过程如下:
代码如下:
def insert_sort(a):
n = len(a)
for i in range(1, n):
cur_val = a[i]
pos = i
while pos > 0 and a[pos - 1] > cur_val:
a[pos] = a[pos - 1]
pos -= 1
a[pos] = cur_val
return a
if __name__ == "__main__":
a = [90, 5, 83, 42, 12, 15]
print(insert_sort(a))
希尔排序
希尔排序是插入排序的另一种变化,通过有间隔的插入排序并逐步减小间隔实现最终的排序,其过程如下:
代码如下:
def shell_sort(a):
n = len(a)
gap = n // 2
while gap > 0:
for i in range(gap):
gap_insert(a, i, gap) #有间隔的插入排序
gap //= 2
return a
def gap_insert(a, sta, gap):
for i in range(sta + gap, len(a), gap):
cur_val = a[i]
pos = i
while pos > sta and a[pos - gap] > cur_val:
a[pos] = a[pos - gap]
pos -= gap
a[pos] = cur_val
return a
if __name__ == "__main__":
a = [90, 5, 83, 42, 12, 15]
print(shell_sort(a))
NB三人组
归并排序
归并排序是基于分治的思想,将需要排序的数据分为两个子序列,对子序列进行排序,然后将排好序的子序列进行合并实现最终的排序,过程如下:
代码如下:
def merge_sort(a):
if len(a) <= 1:
return a
n = len(a) // 2
left = merge_sort(a[:n]) #子序列归并排序
right = merge_sort(a[n:])
return merge(left, right) #合并排好序的子序列
def merge(left, right):
l, r = 0, 0
res = []
while l < len(left) and r < len(right):
if left[l] < right[r]:
res.append(left[l])
l += 1
else:
res.append(right[r])
r += 1
res.extend(left[l:])
res.extend(right[r:])
return res
if __name__ == "__main__":
a = [90, 5, 83, 42, 12, 15]
print(merge_sort(a))
堆排序
堆排序思想是建立一个大根堆,将堆顶位置与最后一个进行交换,再建立大根堆,重复上述操作实现排序,过程如下:
代码如下:
def heap_sort(a):
n = len(a)
for i in range(n // 2 - 1, -1, -1):
siftdown(a, i, n - 1) #建立大根堆
for j in range(n - 1, 0, -1):
a[0], a[j] = a[j], a[0] #交换后,继续建立大根堆
siftdown(a, 0, j-1)
return a
def siftdown(a, sta, end):
root = sta #根节点
while True:
child = 2 * root + 1 #左孩子节点
if child > end:
break
if child + 1 <= end and a[child] < a[child + 1]: #存在右孩子节点
child += 1
if a[root] < a[child]: #维护大根堆
a[root], a[child] = a[child], a[root]
root = child
else:
break
return a
if __name__ == "__main__":
a = [90, 5, 83, 42, 12, 15]
print(heap_sort(a))
快速排序
快速排序的思想是选择一个基准线,将比基准线小的放在一边,比基准线大的放在另一边,通过对两部分进行排序实现最终的排序,过程如下:
代码如下:
def quick_sort(a):
if len(a) <= 1:
return a
left = []
right = []
base = a.pop()
for x in a:
if x < base:
left.append(x)
else:
right.append(x)
return quick_sort(left) + [base] + quick_sort(right)
if __name__ == "__main__":
a = [90, 5, 83, 42, 12, 15]
print(quick_sort(a))
线性时间排序
计数排序
计数排序的思想是建立计数器,统计每个数字出现的次数,再将统计的结果输出实现最终的排序,过程如下:
代码如下:
def count_sort(a):
n = len(a)
max_val = max(a)
count = [0] * (max_val + 1)
for i in range(n):
count[a[i]] += 1
res = []
for i in range(max_val + 1):
for j in range(count[i]):
res.append(i)
return res
if __name__ == "__main__":
a = [90, 5, 83, 42, 12, 15]
print(count_sort(a))
桶排序
桶排序的思想就是将对应范围内的元素放进桶中,对桶中的元素进行排序,然后再将元素按照顺序取出,完成最终的排序,过程如下:
代码如下:
def bucket_sort(a,n=100,max_num=10000):
buckets = [[] for _ in range(n)] #创建桶
for x in a:
i = min(x // (max_num // n) , n - 1)
buckets[i].append(x) #将对应的数据放进桶中
for j in range(len(buckets[i]) - 1 , 0 ,-1):
if buckets[i][j] < buckets[i][j - 1]:
buckets[i][j] , buckets[i][j - 1] = buckets[i][j - 1] , buckets[i][j]
else:
break
result = []
for bin in buckets:
result.extend(bin)
return result
if __name__ == "__main__":
a = [90, 5, 83, 42, 12, 15]
print(bucket_sort(a))
基数排序
基数排序的思想就是将整数按位分为不同的数字,对每个数字进行比较,具体过程如下:
代码如下:
def radix_sort(a):
max_val = max(a)
it = 0
while 10 ** it <= max_val:
buckets = [[] for _ in range(10)]
for x in a:
i = (x // (10 ** it)) % 10
buckets[i].append(x)
a = [j for i in buckets for j in i]
it += 1
return a
if __name__ == "__main__":
a = [90, 5, 83, 42, 12, 15]
print(radix_sort(a))
- 上一篇: 「图解算法数据结构」时间复杂度定义
- 下一篇: 排序算法1|冒泡排序(比较、交换类)(附动图)
猜你喜欢
- 2024-11-10 时间复杂度 时间复杂度取决于
- 2024-11-10 选择排序代码及时间空间复杂度 简单选择排序时间复杂度分析
- 2024-11-10 排序算法汇总 排序算法 简书
- 2024-11-10 「时间管理」JavaScript算法时间、空间复杂度分析
- 2024-11-10 数据结构与算法——常见排序算法分享
- 2024-11-10 数据结构与算法-排序(八)计数排序(Counting Sort)
- 2024-11-10 快速排序算法 快速排序算法的平均时间复杂度为
- 2024-11-10 上个厕所的功夫,就学会了“快速排序”算法
- 2024-11-10 常用排序方法使用场景和性能对比分析
- 2024-11-10 数据结构:复杂度分析(时间复杂度和空间复杂度)
你 发表评论:
欢迎- 最近发表
- 标签列表
-
- oraclesql优化 (66)
- 类的加载机制 (75)
- feignclient (62)
- 一致性hash算法 (71)
- dockfile (66)
- 锁机制 (57)
- javaresponse (60)
- 查看hive版本 (59)
- phpworkerman (57)
- spark算子 (58)
- vue双向绑定的原理 (68)
- springbootget请求 (58)
- docker网络三种模式 (67)
- spring控制反转 (71)
- data:image/jpeg (69)
- base64 (69)
- java分页 (64)
- kibanadocker (60)
- qabstracttablemodel (62)
- java生成pdf文件 (69)
- deletelater (62)
- com.aspose.words (58)
- android.mk (62)
- qopengl (73)
- epoch_millis (61)
本文暂时没有评论,来添加一个吧(●'◡'●)