经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 程序设计 » Python3 » 查看文章
python3实现常见的排序算法(示例代码)
来源:jb51  时间:2021/7/5 8:34:04  对本文有异议

冒泡排序

冒泡排序是一种简单的排序算法。它重复地走访过要排序的数列,一次比较两个元素,如果它们的顺序错误就把它们交换过来。走访数列的工作是重复地进行直到没有再需要交换,也就是说该数列已经排序完成。这个算法的名字由来是因为越小的元素会经由交换慢慢“浮”到数列的顶端。

  1. def mao(lst):
  2. for i in range(len(lst)):
  3. # 由于每一轮结束后,总一定有一个大的数排在后面
  4. # 而且后面的数已经排好了
  5. # 即i轮之后,就有i个数字被排好
  6. # 所以其 len-1 -i到 len-1的位置是已经排好的了
  7. # 只需要比较0到len -1 -i的位置即可
  8.  
  9. # flag 用于标记是否刚开始就是排好的数据
  10. # 只有当flag状态发生改变时(第一次循环就可以确定),继续排序,否则返回
  11. flag = False
  12. for j in range(len(lst) - i - 1):
  13. if lst[j] > lst[j + 1]:
  14. lst[j], lst[j + 1] = lst[j + 1], lst[j]
  15. flag = True
  16. # 非排好的数据,改变flag
  17. if not flag:
  18. return lst
  19. return lst
  20.  
  21. print(mao([1, 5, 55, 4, 5, 1, 3, 4, 5, 8, 62, 7]))

选择排序

选择排序是一种简单直观的排序算法。它的工作原理:首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾。以此类推,直到所有元素均排序完毕。

  1. # 选择排序是从前开始排的
  2. # 选择排序是从一个列表中找出一个最小的元素,然后放在第一位上。
  3. # 冒泡排序类似
  4. # 其 0 到 i的位置是排好的,只需要排i+1到len(lst)-1即可
  5.  
  6. def select_sort(lst):
  7. for i in range(len(lst)):
  8. min_index = i # 用于记录最小的元素的索引
  9. for j in range(i + 1, len(lst)):
  10. if lst[j] < lst[min_index]:
  11. min_index = j
  12.  
  13. # 此时,已经确定,min_index为 i+1 到len(lst) - 1 这个区间最小值的索引
  14. lst[i], lst[min_index] = lst[min_index], lst[i]
  15.  
  16. return lst
  17.  
  18.  
  19. def select_sort2(lst):
  20. # 第二种选择排序的方法
  21. # 原理与第一种一样
  22. # 不过不需要引用中间变量min_index
  23. # 只需要找到索引i后面的i+1到len(lst)的元素即可
  24.  
  25. for i in range(len(lst)):
  26. for j in range(len(lst) - i):
  27.  
  28. # lst[i + j]是一个i到len(lst)-1的一个数
  29. # 因为j <= len(lst) -i 即 j + i <= len(lst)
  30. if lst[i] > lst[i + j]:
  31. # 说明后面的数更小,更换位置
  32. lst[i], lst[i + j] = lst[i + j], lst[i]
  33. return lst
  34.  
  35.  
  36. print(select_sort([1, 5, 55, 4, 5, 1, 3, 4, 5, 8, 62, 7]))
  37. print(select_sort2([1, 5, 55, 4, 5, 1, 3, 4, 5, 8, 62, 7]))

快速排序

快速排序是通过一趟排序将待排记录分隔成独立的两部分,其中一部分记录的关键字均比另一部分的关键字小,则可分别对这两部分记录继续进行排序,以达到整个序列有序。

  1. # 原理
  2. # 1. 任取列表中的一个元素i
  3. # 2. 把列表中大于i的元素放于其右边,小于则放于其左边
  4. # 3. 如此重复
  5. # 4. 直到不能在分,即只剩1个元素了
  6. # 5. 然后将这些结果组合起来
  7.  
  8. def quicksort(lst):
  9. if len(lst) < 2: # lst有可能为空
  10. return lst
  11.  
  12. # ['pɪvət] 中心点
  13. pivot = lst[0]
  14. less_lst = [i for i in lst[1:] if i <= pivot]
  15. greater_lst = [i for i in lst[1:] if i > pivot]
  16. # 最后的结果就是
  17. # 左边的结果 + 中间值 + 右边的结果
  18. # 然后细分 左+中+右 + 中间值 + 左 + 中+ 右
  19. # ........... + 中间值 + ............
  20. return quicksort(less_lst) + [pivot] + quicksort(greater_lst)
  21.  
  22.  
  23. print(quicksort([1, 5, 55, 4, 5, 1, 3, 4, 5, 8, 62, 7]))
  24. print(quicksort([1, 5, 8, 62]))

插入排序

插入排序的算法描述是一种简单直观的排序算法。它的工作原理是通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应位置并插入。

  1. # lst的[0, i) 项是有序的,因为已经排过了
  2. # 那么只需要比对第i项的lst[i]和lst[0 : i]的元素大小即可
  3. # 假如,lst[i]大,则不用改变位置
  4. # 否则,lst[i]改变位置,插到j的位置,而lst[j]自然往后挪一位
  5. # 然后再删除lst[i+1]即可(lst[i+1]是原来的lst[i])
  6. #
  7. # 重复上面步骤即可,排序完成
  8.  
  9. def insert_sort(lst: list):
  10. # 外层开始的位置从1开始,因为从0开始都没得排
  11. # 只有两个元素以上才能排序
  12. for i in range(1, len(lst)):
  13. # 内层需要从0开始,因为lst[0]的位置不一定是最小的
  14. for j in range(i):
  15. if lst[i] < lst[j]:
  16. lst.insert(j, lst[i])
  17. # lst[i]已经插入到j的位置了,j之后的元素都往后+1位,所以删除lst[i+1]
  18. del lst[i + 1]
  19. return lst
  20.  
  21. print(insert_sort([1, 5, 55, 4, 5, 1, 3, 4, 5, 8, 62, 7]))

希尔排序

希尔排序是1959年Shell发明的,第一个突破O(n2)的排序算法,是简单插入排序的改进版。它与插入排序的不同之处在于,它会优先比较距离较远的元素。希尔排序又叫缩小增量排序。

希尔排序

  1. # 希尔排序是对直接插入排序的优化版本
  2. # 1. 分组:
  3. # 每间隔一段距离取一个元素为一组
  4. # 间隔自己确定,一般为lst的一半
  5. # 2. 根据插入排序,把每一组排序好
  6. # 3. 继续分组:
  7. # 同样没间隔一段距离取一个元素为一组
  8. # 间隔要求比 之前的间隔少一半
  9. # 4. 再每组插入排序
  10. # 5. 直到间隔为1,则排序完成
  11. #
  12.  
  13. def shell_sort(lst):
  14. lst_len = len(lst)
  15. gap = lst_len // 2 # 整除2,取间隔
  16. while gap >= 1: # 间隔为0时结束
  17. for i in range(gap, lst_len):
  18. temp = lst[i]
  19. j = i
  20. # 插入排序
  21. while j - gap >= 0 and lst[j - gap] > temp:
  22. lst[j] = lst[j - gap]
  23. j -= gap
  24. lst[j] = temp
  25. gap //= 2
  26. return lst
  27.  
  28.  
  29. print(shell_sort([1, 5, 55, 4, 5, 1, 3, 4, 5, 8, 62, 7]))
  30.  
  31.  
  32. # 奇数
  33. # gap = 2
  34. # [5, 2, 4, 3, 1]
  35. # [5, 4, 1] [2, 3]
  36. # [1, 4, 5, 2, 3]
  37. # gap = 1
  38. # [1, 2, 3, 4, 5]
  39.  
  40. # 偶数
  41. # gap = 3
  42. # [5, 2, 4, 3, 1, 6]
  43. # [5, 3] [2, 1] [4,6]
  44. # [3, 5, 1, 2, 4 , 6]
  45. # gap = 1
  46. # [1, 2, 3, 4, 5, 6]

并归排序

归并排序是建立在归并操作上的一种有效的排序算法。该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。若将两个有序表合并成一个有序表,称为2-路归并。

并归排序

  1. # 利用分治法
  2. # 不断将lst分为左右两个分
  3. # 直到不能再分
  4. # 然后返回
  5. # 将两边的列表的元素进行比对,排序然后返回
  6. # 不断重复上面这一步骤
  7. # 直到排序完成,即两个大的列表比对完成
  8.  
  9.  
  10. def merge(left, right):
  11. # left 可能为只有一个元素的列表,或已经排好序的多个元素列表(之前调用过merge)
  12. # right 也一样
  13.  
  14. res = []
  15. while left and right:
  16. item = left.pop(0) if left[0] < right[0] else right.pop(0)
  17. res.append(item)
  18.  
  19. # 此时,left或right已经有一个为空,直接extend插入
  20. # 而且,left和right是之前已经排好序的列表,不需要再操作了
  21.  
  22. res.extend(left)
  23. res.extend(right)
  24. return res
  25.  
  26.  
  27. def merge_sort(lst):
  28. lst_len = len(lst)
  29. if lst_len <= 1:
  30. return lst
  31. mid = lst_len // 2
  32.  
  33. lst_right = merge_sort(lst[mid:len(lst)]) # 返回的时lst_len <= 1时的 lst 或 merge中进行排序后的列表
  34. lst_left = merge_sort(lst[:mid]) # 返回的是lst_len <= 1时的 lst 或 merge中进行排序后的列表
  35.  
  36. return merge(lst_left, lst_right) # 进行排序,lst_left lst_right 的元素会不断增加
  37.  
  38.  
  39. print(merge_sort([1, 5, 55, 4, 5, 1, 3, 4, 5, 8, 62, 7]))

堆排序

堆排序是指利用堆这种数据结构所设计的一种排序算法。堆积是一个近似完全二叉树的结构,并同时满足堆积的性质:即子结点的键值或索引总是小于(或者大于)它的父节点。然后进行排序。

堆排序

  1. # 把列表创成一个大根堆或小根堆
  2. # 然后根据大(小)根堆的特点:根节点最大(小),逐一取值
  3. #
  4. # 升序----使用大顶堆
  5. #
  6. # 降序----使用小顶堆
  7. # 本例以小根堆为例
  8. # 列表lst = [1, 22 ,11, 8, 12, 4, 9]
  9.  
  10. # 1. 建成一个普通的堆:
  11. # 1
  12. # / # 22 11
  13. # / \ / # 8 12 4 9
  14. #
  15. # 2. 进行调整,从子开始调整位置,要求: 父节点<= 字节点
  16. #
  17. # 1 1 1
  18. # / \ 13和22调换位置 / \ 4和11调换位置 / # 22 11 ==============> 13 11 ==============> 13 4
  19. # / \ / \ / \ / \ / \ / # 13 14 4 9 22 14 4 9 22 14 11 9
  20. #
  21. # 3. 取出树上根节点,即最小值,把换上叶子节点的最大值
  22. #
  23. # 1
  24. # /
  25. # ~~~~/
  26. # 22
  27. # / # 8 4
  28. # \ / # 12 11 9
  29. #
  30. # 4. 按照同样的道理,继续形成小根堆,然后取出根节点,。。。。重复这个过程
  31. #
  32. # 1 1 1 4 1 4 1 4 8 1 4 8
  33. # / / / / / /
  34. # ~~~/ ~~~/ ~~~/ ~~~/ ~~~/ ~~~/
  35. # 22 4 22 8 22 9
  36. # / \ / \ / \ / \ / \ / # 8 4 8 9 8 9 12 9 12 9 12 11
  37. # \ / \ \ / \ \ / \ / / /
  38. # 12 11 9 12 11 22 12 11 22 11 11 22
  39. #
  40. # 续上:
  41. # 1 4 8 9 1 4 8 9 1 4 8 9 11 1 4 8 9 11 1 4 8 9 11 12 ==> 1 4 8 9 11 12 22
  42. # / / / / /
  43. # ~~~/ ~~~/ ~~~/ ~~~/ ~~~/
  44. # 22 11 22 12 22
  45. # / \ / \ / /
  46. # 12 11 12 22 12 22
  47. #
  48. # 代码实现
  49.  
  50. def heapify(lst, lst_len, i):
  51. """创建一个堆"""
  52. less = i # largest为最大元素的索引
  53.  
  54. left_node_index = 2 * i + 1 # 左子节点索引
  55. right_node_index = 2 * i + 2 # 右子节点索引
  56.  
  57. # lst[i] 就是父节点(假如有子节点的话):
  58. #
  59. # lst[i]
  60. # / # lst[2*i + 1] lst[ 2*i + 2]
  61. #
  62.  
  63. # 想要大根堆,即升序, 将判断左右子节点大小的 ‘>' 改为 ‘<' 即可
  64. #
  65. if left_node_index < lst_len and lst[less] > lst[left_node_index]:
  66. less = left_node_index
  67.  
  68. if right_node_index < lst_len and lst[less] > lst[right_node_index]:
  69. # 右边节点最小的时候
  70. less = right_node_index
  71.  
  72. if less != i:
  73. # 此时,是字节点大于父节点,所以相互交换位置
  74. lst[i], lst[less] = lst[less], lst[i] # 交换
  75. heapify(lst, lst_len, less)
  76. # 节点变动了,需要再检查一下
  77.  
  78. def heap_sort(lst):
  79. res = []
  80. i = len(lst)
  81. lst_len = len(lst)
  82.  
  83. for i in range(lst_len, -1, -1):
  84. # 要从叶节点开始比较,所以倒着来
  85. heapify(lst, lst_len, i)
  86.  
  87. # 此时,已经建好了一个小根树
  88. # 所以,交换元素,将根节点(最小值)放在后面,重复这个过程
  89. for j in range(lst_len - 1, 0, -1):
  90. lst[0], lst[j] = lst[j], lst[0] # 交换,最小的放在j的位置
  91.  
  92. heapify(lst, j, 0) # 再次构建一个[0: j)小根堆 [j: lst_len-1]已经倒序排好了
  93. return lst
  94.  
  95. arr = [1, 5, 55, 4, 5, 1, 3, 4, 5, 8, 62, 7]
  96. print(heap_sort(arr))

参考:
十大经典排序算法(动图演示)
数据结构与算法-排序篇-Python描述

动图可以点击这里查看

我的github
我的博客
我的笔记

到此这篇关于python3实现常见的排序算法(示例代码)的文章就介绍到这了,更多相关python排序算法内容请搜索w3xue以前的文章或继续浏览下面的相关文章希望大家以后多多支持w3xue!

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号