快速排序
(1)基本概念
快速排序(Quicksort),又称划分交换排序(partition-exchange sort)。它的基本思想是分治法,通过选择一个基准元素,将数组分为两部分,一部分小于基准元素,另一部分大于基准元素,然后递归地对这两部分进行快速排序(递归)。
快速排序也是递归算法,其递归三要素为:
基本结束条件:数据表仅有一个数据项且是排好序的。
缩小规模:根据基准元素将数据表分为两半
调用自身:将两半分别调用自身进行排序(排序的基本操作在分裂过程中)
分裂数据表的手段是设置左右标,左标向右移动(遇到比基准元素大的就停止),右标向左移动(遇到比基准元素小的就停止),然后把左右标所指的数据项进行交换,继续移动,直到左标移到右标的右侧,停止移动,将基准元素的位置与右标的位置进行交换,分裂完成。左半部比基准值小,右半部比基准值大。
参考链接:http://t.csdnimg.cn/JPbuo
(2)代码示例
def quick_sort(alist, start, end):#alist是待排序列表
"""快速排序"""
if start >= end: # 递归的退出条件
return
mid = alist[start] # 设定起始的基准元素
#初始化指针
low = start # low为序列左边在开始位置的由左向右移动的游标
high = end # high为序列右边末尾位置的由右向左移动的游标
while low < high:#将列表由基准元素分为左右两部分
while low < high and alist[high] >= mid:#如果low与high未重合,high(右边)指向的元素大于等于基准元素
high -= 1 #high向左移动
alist[low] = alist[high] #此时high指向的元素小于基准元素,将high指向的元素的值赋值给low指向的位置
while low < high and alist[low] < mid: # 如果low与high未重合,low指向的元素比基准元素小
low += 1#low向右移动
alist[high] = alist[low] # 此时low指向一个比基准元素大的元素,将low指向的元素放到high空着的位置上
# 退出循环后,low与high重合,此时所指位置为基准元素的正确位置,左边的元素都比基准元素小,右边的元素都比基准元素大
alist[low] = mid # 将基准元素放到该位置
#进行递归调用
# 对基准元素左边的子序列进行快速排序
quick_sort(alist, start, low - 1) # start :0 low -1 原基准元素靠左边一位
# 对基准元素右边的子序列进行快速排序
quick_sort(alist, low + 1, end) # low+1 : 原基准元素靠右一位 end: 最后
if __name__ == '__main__':
alist = [54, 26, 93, 17, 77, 31, 44, 55, 20]
quick_sort(alist, 0, len(alist) - 1)
print(alist)
(3)算法分析
空间上:由于是对原数组进行就地排序,因此空间复杂度为O(1)
最优时间复杂度:O(nlogn)
最坏时间复杂度:O(n2):需要排序的数组是降序或者升序排序的
稳定性:不稳定,容易受数据怎么排序的影响,数据改变了,排序的过程会改变,时间也会改变
在最好的情况,每次我们运行一次分区,我们会把一个数列分为两个几近相等的片段。这个意思就是每次递归调用处理一半大小的数列。因此,在到达大小为一的数列前,我们只要作log n次嵌套的调用。这个意思就是调用树的深度是O(log n),则总体的最优时间复杂度为:O(nlogn)