Python实现的数据结构与算法之快速排序详解


Posted in Python onApril 22, 2015

本文实例讲述了Python实现的数据结构与算法之快速排序。分享给大家供大家参考。具体分析如下:

一、概述

快速排序(quick sort)是一种分治排序算法。该算法首先 选取 一个划分元素(partition element,有时又称为pivot);接着重排列表将其 划分 为三个部分:left(小于划分元素pivot的部分)、划分元素pivot、right(大于划分元素pivot的部分),此时,划分元素pivot已经在列表的最终位置上;然后分别对left和right两个部分进行 递归排序

其中,划分元素的 选取 直接影响到快速排序算法的效率,通常选择列表的第一个元素或者中间元素或者最后一个元素作为划分元素,当然也有更复杂的选择方式;划分 过程根据划分元素重排列表,是快速排序算法的关键所在,该过程的原理示意图如下:

<-- 选取划分元素 -->

Python实现的数据结构与算法之快速排序详解

<-- 划分过程 -->

Python实现的数据结构与算法之快速排序详解

<-- 划分结果 -->

Python实现的数据结构与算法之快速排序详解

快速排序算法的优点是:原位排序(只使用很小的辅助栈),平均情况下的时间复杂度为 O(n log n)。快速排序算法的缺点是:它是不稳定的排序算法,最坏情况下的时间复杂度为 O(n2)。

二、Python实现

1、标准实现

#!/usr/bin/env python
# -*- coding: utf-8 -*-
def stdQuicksort(L):
  qsort(L, 0, len(L) - 1)
def qsort(L, first, last):
  if first < last:
    split = partition(L, first, last)
    qsort(L, first, split - 1)
    qsort(L, split + 1, last)
def partition(L, first, last):
  # 选取列表中的第一个元素作为划分元素
  pivot = L[first]
  leftmark = first + 1
  rightmark = last
  while True:
    while L[leftmark] <= pivot: 
 # 如果列表中存在与划分元素pivot相等的元素,让它位于left部分
     # 以下检测用于划分元素pivot是列表中的最大元素时,
  #防止leftmark越界
      if leftmark == rightmark:
        break
      leftmark += 1
    while L[rightmark] > pivot:
      # 这里不需要检测,划分元素pivot是列表中的最小元素时,
      # rightmark会自动停在first处
      rightmark -= 1
    if leftmark < rightmark:
      # 此时,leftmark处的元素大于pivot,
   #而rightmark处的元素小于等于pivot,交换二者
      L[leftmark], L[rightmark] = L[rightmark], L[leftmark]
    else:
      break
  # 交换first处的划分元素与rightmark处的元素
  L[first], L[rightmark] = L[rightmark], L[first]
  # 返回划分元素pivot的最终位置
  return rightmark

2、Pythonic实现

#!/usr/bin/env python
# -*- coding: utf-8 -*-
def pycQuicksort(L):
  if len(L) <= 1: return L
  return pycQuicksort([x for x in L if x < L[0]]) + \
      [x for x in L if x == L[0]] + \
      pycQuicksort([x for x in L if x > L[0]])

对比 标准实现 可以看出,Pythonic实现 更简洁、更直观、更酷。但需要指出的是,Pythonic实现 使用了Python中的 列表解析 (List Comprehension,也叫列表展开、列表推导),每一次 递归排序 都会产生新的列表,因此失去了快速排序算法本来的 原位排序 的优点。

三、算法测试

#!/usr/bin/env python
# -*- coding: utf-8 -*-
if __name__ == '__main__':
  L = [54, 26, 93, 17, 77, 31, 44, 55, 20]
  M = L[:]
  print('before stdQuicksort: ' + str(L))
  stdQuicksort(L)
  print('after stdQuicksort: ' + str(L))
  print('before pycQuicksort: ' + str(M))
  print('after pycQuicksort: ' + str(pycQuicksort(M)))

运行结果:

$ python testquicksort.py
before stdQuicksort: [54, 26, 93, 17, 77, 31, 44, 55, 20]
after stdQuicksort: [17, 20, 26, 31, 44, 54, 55, 77, 93]
before pycQuicksort: [54, 26, 93, 17, 77, 31, 44, 55, 20]
after pycQuicksort: [17, 20, 26, 31, 44, 54, 55, 77, 93]

希望本文所述对大家的Python程序设计有所帮助。

Python 相关文章推荐
python 图片验证码代码分享
Jul 04 Python
Python构造函数及解构函数介绍
Feb 26 Python
用Python制作简单的朴素基数估计器的教程
Apr 01 Python
在Python的一段程序中如何使用多次事件循环详解
Sep 07 Python
Python中存取文件的4种不同操作
Jul 02 Python
Python实现FTP文件传输的实例
Jul 07 Python
Python代码实现http/https代理服务器的脚本
Aug 12 Python
Jupyter Notebook折叠输出的内容实例
Apr 22 Python
windows10在visual studio2019下配置使用openCV4.3.0
Jul 14 Python
python爬取微博评论的实例讲解
Jan 15 Python
Pytorch模型迁移和迁移学习,导入部分模型参数的操作
Mar 03 Python
 分享一个Python 遇到数据库超好用的模块
Apr 06 Python
利用Fn.py库在Python中进行函数式编程
Apr 22 #Python
Python实现的数据结构与算法之基本搜索详解
Apr 22 #Python
Python实现的数据结构与算法之链表详解
Apr 22 #Python
Python实现的数据结构与算法之双端队列详解
Apr 22 #Python
Python实现的数据结构与算法之队列详解
Apr 22 #Python
详尽讲述用Python的Django框架测试驱动开发的教程
Apr 22 #Python
Hadoop中的Python框架的使用指南
Apr 22 #Python
You might like
德劲1103二次变频版的打磨
2021/03/02 无线电
使PHP自定义函数返回多个值
2006/11/26 PHP
PHP 防注入函数(格式化数据)
2011/08/08 PHP
php批量添加数据与批量更新数据的实现方法
2014/12/16 PHP
thinkPHP3.x常量整理(预定义常量/路径常量/系统常量)
2016/05/20 PHP
javascript 面向对象编程 万物皆对象
2009/09/17 Javascript
JavaScript OOP面向对象介绍
2010/12/02 Javascript
img onload事件绑定各浏览器均可执行
2012/12/19 Javascript
根据选择不同的下拉值出现相对应的文本输入框
2013/08/01 Javascript
JS 实现BASE64_ENCODE和BASE64_DECODE(实例代码)
2013/11/13 Javascript
jquery 合并内容相同的单元格(示例代码)
2013/12/13 Javascript
PHP中CURL的几个经典应用实例
2015/01/23 Javascript
js全选实现和判断是否有复选框选中的方法
2015/02/17 Javascript
JS实现网页顶部向下滑出的全国城市切换导航效果
2015/08/22 Javascript
Actionscript与javascript交互实例程序(修改)
2016/09/22 Javascript
JavaScript实现图片切换效果
2017/08/12 Javascript
JS面向对象的程序设计相关知识小结
2018/05/26 Javascript
vue动态设置img的src路径实例
2018/09/18 Javascript
vue实现的双向数据绑定操作示例
2018/12/04 Javascript
使用localStorage替代cookie做本地存储
2019/09/25 Javascript
vue 翻页组件vue-flip-page效果
2020/02/05 Javascript
[01:04:09]DOTA2-DPC中国联赛 正赛 iG vs VG BO3 第二场 2月2日
2021/03/11 DOTA
Python 操作文件的基本方法总结
2017/08/10 Python
利用python爬取斗鱼app中照片方法实例
2017/12/03 Python
python matplotlib绘图,修改坐标轴刻度为文字的实例
2018/05/25 Python
Python使用pickle模块实现序列化功能示例
2018/07/13 Python
使用Python将字符串转换为格式化的日期时间字符串
2019/09/01 Python
python如何获取apk的packagename和activity
2020/01/10 Python
python退出循环的方法
2020/06/18 Python
Django URL参数Template反向解析
2020/11/24 Python
Strathberry苏贝瑞中国官网:西班牙高级工匠手工打造
2020/10/19 全球购物
机械专业应届生求职信
2013/09/21 职场文书
卖房协议书
2014/04/11 职场文书
大学生考试作弊检讨书1000字
2014/10/14 职场文书
环保证明
2015/06/23 职场文书
Vue2.0搭建脚手架
2022/03/13 Vue.js