python實現(xiàn)八大排序算法(2)
本文接上一篇博客python實現(xiàn)的八大排序算法part1,將繼續(xù)使用python實現(xiàn)八大排序算法中的剩余四個:快速排序、堆排序、歸并排序、基數(shù)排序
5、快速排序
快速排序是通常被認為在同數(shù)量級(O(nlog2n))的排序方法中平均性能最好的。
算法思想:
已知一組無序數(shù)據(jù)a[1]、a[2]、……a[n],需將其按升序排列。首先任取數(shù)據(jù)a[x]作為基準。比較a[x]與其它數(shù)據(jù)并排序,使a[x]排在數(shù)據(jù)的第k位,并且使a[1]~a[k-1]中的每一個數(shù)據(jù)<a[x],a[k+1]~a[n]中的每一個數(shù)據(jù)>a[x],然后采用分治的策略分別對a[1]~a[k-1]和a[k+1]~a[n]兩組數(shù)據(jù)進行快速排序。
優(yōu)點:極快,數(shù)據(jù)移動少;
缺點:不穩(wěn)定。
python代碼實現(xiàn):
def quick_sort(list): little = [] pivotList = [] large = [] # 遞歸出口 if len(list) <= 1: return list else: # 將第一個值做為基準 pivot = list[0] for i in list: # 將比基準小的值放到less數(shù)列 if i < pivot: little.append(i) # 將比基準大的值放到more數(shù)列 elif i > pivot: large.append(i) # 將和基準相同的值保存在基準數(shù)列 else: pivotList.append(i) # 對less數(shù)列和more數(shù)列繼續(xù)進行快速排序 little = quick_sort(little) large = quick_sort(large) return little + pivotList + large
下面這段代碼出自《Python cookbook 第二版的三行實現(xiàn)python快速排序。
#!/usr/bin/env python #coding:utf-8 ''' file:python-8sort.py date:9/1/17 9:03 AM author:lockey email:lockey@123.com desc:python實現(xiàn)八大排序算法 ''' lst = [65,568,9,23,4,34,65,8,6,9] def quick_sort(list): if len(list) <= 1: return list else: pivot = list[0] return quick_sort([x for x in list[1:] if x < pivot]) + \ [pivot] + \ quick_sort([x for x in list[1:] if x >= pivot])
運行測試結(jié)果截圖:
好吧,還有更精簡的語法糖,一行完事:
quick_sort = lambda xs : ( (len(xs) <= 1 and [xs]) or [ quick_sort( [x for x in xs[1:] if x < xs[0]] ) + [xs[0]] + quick_sort( [x for x in xs[1:] if x >= xs[0]] ) ] )[0]
若初始序列按關(guān)鍵碼有序或基本有序時,快排序反而蛻化為冒泡排序。為改進之,通常以“三者取中法”來選取基準記錄,即將排序區(qū)間的兩個端點與中點三個記錄關(guān)鍵碼居中的調(diào)整為支點記錄??焖倥判蚴且粋€不穩(wěn)定的排序方法。
在改進算法中,我們將只對長度大于k的子序列遞歸調(diào)用快速排序,讓原序列基本有序,然后再對整個基本有序序列用插入排序算法排序。實踐證明,改進后的算法時間復(fù)雜度有所降低,且當k取值為 8 左右時,改進算法的性能最佳。
6、堆排序(Heap Sort)
堆排序是一種樹形選擇排序,是對直接選擇排序的有效改進。
優(yōu)點 : 效率高
缺點:不穩(wěn)定
堆的定義下:具有n個元素的序列 (h1,h2,...,hn),當且僅當滿足(hi>=h2i,hi>=2i+1)或(hi<=h2i,hi<=2i+1) (i=1,2,...,n/2)時稱之為堆。在這里只討論滿足前者條件的堆。由堆的定義可以看出,堆頂元素(即第一個元素)必為最大項(大頂堆)。完全二 叉樹可以很直觀地表示堆的結(jié)構(gòu)。堆頂為根,其它為左子樹、右子樹。
算法思想:
初始時把要排序的數(shù)的序列看作是一棵順序存儲的二叉樹,調(diào)整它們的存儲序,使之成為一個 堆,這時堆的根節(jié)點的數(shù)最大。然后將根節(jié)點與堆的最后一個節(jié)點交換。然后對前面(n-1)個數(shù)重新調(diào)整使之成為堆。依此類推,直到只有兩個節(jié)點的堆,并對 它們作交換,最后得到有n個節(jié)點的有序序列。從算法描述來看,堆排序需要兩個過程,一是建立堆,二是堆頂與堆的最后一個元素交換位置。所以堆排序有兩個函數(shù)組成。一是建堆的滲透函數(shù),二是反復(fù)調(diào)用滲透函數(shù)實現(xiàn)排序的函數(shù)。
python代碼實現(xiàn):
# -*- coding: UTF-8 -*- ''' Created on 2017年9月2日 Running environment:win7.x86_64 eclipse python3 @author: Lockey ''' lst = [65,568,9,23,4,34,65,8,6,9] def adjust_heap(lists, i, size):# 調(diào)整堆 lchild = 2 * i + 1;rchild = 2 * i + 2 max = i if i < size / 2: if lchild < size and lists[lchild] > lists[max]: max = lchild if rchild < size and lists[rchild] > lists[max]: max = rchild if max != i: lists[max], lists[i] = lists[i], lists[max] adjust_heap(lists, max, size) def build_heap(lists, size):# 創(chuàng)建堆 halfsize = int(size/2) for i in range(0, halfsize)[::-1]: adjust_heap(lists, i, size) def heap_sort(lists):# 堆排序 size = len(lists) build_heap(lists, size) for i in range(0, size)[::-1]: lists[0], lists[i] = lists[i], lists[0] adjust_heap(lists, 0, i) print(lists)
結(jié)果示例:
7、歸并排序
算法思想:
歸并(Merge)排序是建立在歸并操作上的一種有效的排序算法,該算法是采用分治法的一個非常典型的應(yīng)用。將已有序的子序列合并,得到完全有序的序列;即先使每個子序列有序,再使子序列段間有序。若將兩個有序表合并成一個有序表,稱為二路歸并。
歸并過程為:比較a[i]和a[j]的大小,若a[i]≤a[j],則將第一個有序表中的元素a[i]復(fù)制到r[k]中,并令i和k分別加上1;否則將第二個有序表中的元素a[j]復(fù)制到r[k]中,并令j和k分別加上1,如此循環(huán)下去,直到其中一個有序表取完,然后再將另一個有序表中剩余的元素復(fù)制到r中從下標k到下標t的單元。歸并排序的算法我們通常用遞歸實現(xiàn),先把待排序區(qū)間[s,t]以中點二分,接著把左邊子區(qū)間排序,再把右邊子區(qū)間排序,最后把左區(qū)間和右區(qū)間用一次歸并操作合并成有序的區(qū)間[s,t]。
# -*- coding: UTF-8 -*- ''' Created on 2017年9月2日 Running environment:win7.x86_64 eclipse python3 @author: Lockey ''' lst = [65,568,9,23,4,34,65,8,6,9] def merge(left, right): i, j = 0, 0 result = [] while i < len(left) and j < len(right): if left[i] <= right[j]: result.append(left[i]) i += 1 else: result.append(right[j]) j += 1 result += left[i:] result += right[j:] print(result) return result def merge_sort(lists):# 歸并排序 if len(lists) <= 1: return lists num = int(len(lists) / 2) left = merge_sort(lists[:num]) right = merge_sort(lists[num:]) return merge(left, right)
程序結(jié)果示例:
8、桶排序/基數(shù)排序(Radix Sort)
優(yōu)點:快,效率最好能達到O(1)
缺點:
1.首先是空間復(fù)雜度比較高,需要的額外開銷大。排序有兩個數(shù)組的空間開銷,一個存放待排序數(shù)組,一個就是所謂的桶,比如待排序值是從0到m-1,那就需要m個桶,這個桶數(shù)組就要至少m個空間。
2.其次待排序的元素都要在一定的范圍內(nèi)等等。
算法思想:
是將陣列分到有限數(shù)量的桶子里。每個桶子再個別排序(有可能再使用別的排序算法或是以遞回方式繼續(xù)使用桶排序進行排序)。桶排序是鴿巢排序的一種歸納結(jié)果。當要被排序的陣列內(nèi)的數(shù)值是均勻分配的時候,桶排序使用線性時間(Θ(n))。但桶排序并不是 比較排序,他不受到 O(n log n) 下限的影響。
簡單來說,就是把數(shù)據(jù)分組,放在一個個的桶中,然后對每個桶里面的在進行排序。
例如要對大小為[1..1000]范圍內(nèi)的n個整數(shù)A[1..n]排序
首先,可以把桶大小設(shè)為10,這樣就有100個桶了,具體而言,設(shè)集合B[1]存儲[1..10]的整數(shù),集合B[2]存儲 (10..20]的整數(shù),……集合B[i]存儲( (i-1)*10, i*10]的整數(shù),i = 1,2,..100??偣灿?100個桶。
然后,對A[1..n]從頭到尾掃描一遍,把每個A[i]放入對應(yīng)的桶B[j]中。 再對這100個桶中每個桶里的數(shù)字排序,這時可用冒泡,選擇,乃至快排,一般來說任 何排序法都可以。
最后,依次輸出每個桶里面的數(shù)字,且每個桶中的數(shù)字從小到大輸出,這 樣就得到所有數(shù)字排好序的一個序列了。
假設(shè)有n個數(shù)字,有m個桶,如果數(shù)字是平均分布的,則每個桶里面平均有n/m個數(shù)字。如果
對每個桶中的數(shù)字采用快速排序,那么整個算法的復(fù)雜度是
O(n + m * n/m*log(n/m)) = O(n + nlogn - nlogm)
從上式看出,當m接近n的時候,桶排序復(fù)雜度接近O(n)
當然,以上復(fù)雜度的計算是基于輸入的n個數(shù)字是平均分布這個假設(shè)的。這個假設(shè)是很強的 ,實際應(yīng)用中效果并沒有這么好。如果所有的數(shù)字都落在同一個桶中,那就退化成一般的排序了。
python代碼實現(xiàn):
# -*- coding: UTF-8 -*- ''' Created on 2017年9月2日 Running environment:win7.x86_64 eclipse python3 @author: Lockey ''' import math lst = [65,56,9,23,84,34,8,6,9,54,11] #因為列表數(shù)據(jù)范圍在100以內(nèi),所以將使用十個桶來進行排序 def radix_sort(lists, radix=10): k = int(math.ceil(math.log(max(lists), radix))) bucket = [[] for i in range(radix)] for i in range(1, k+1): for j in lists: gg = int(j/(radix**(i-1))) % (radix**i) bucket[gg].append(j) del lists[:] for z in bucket: lists += z del z[:] print(lists) return lists
程序運行測試結(jié)果:
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
python內(nèi)置函數(shù)之slice案例詳解
這篇文章主要介紹了python內(nèi)置函數(shù)之slice案例詳解,本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細內(nèi)容,需要的朋友可以參考下2021-09-09Python求算數(shù)平方根和約數(shù)的方法匯總
這篇文章主要介紹了 Python求算數(shù)平方根和約數(shù)的方法匯總的相關(guān)資料,需要的朋友可以參考下2016-03-03Python時間戳轉(zhuǎn)換為字符串與字符串轉(zhuǎn)換為時間戳
在編寫代碼時,往往涉及時間、日期、時間戳的相互轉(zhuǎn)換,下面這篇文章主要給大家介紹了關(guān)于Python時間戳轉(zhuǎn)換為字符串與字符串轉(zhuǎn)換為時間戳的相關(guān)資料,文中給出了詳細的實例代碼,需要的朋友可以參考下2023-02-02Windows平臺Python連接sqlite3數(shù)據(jù)庫的方法分析
這篇文章主要介紹了Windows平臺Python連接sqlite3數(shù)據(jù)庫的方法,結(jié)合實例形式分析了Windows平臺安裝SQLite數(shù)據(jù)庫及創(chuàng)建、連接數(shù)據(jù)庫的實現(xiàn)方法與相關(guān)注意事項,需要的朋友可以參考下2017-07-07python中subprocess批量執(zhí)行l(wèi)inux命令
本篇文章給大家詳細講述了python中使用subprocess批量執(zhí)行l(wèi)inux命令的方法,有興趣的朋友參考學(xué)習(xí)下。2018-04-04Python-docx 實現(xiàn)整體修改或者部分修改文字的大小和字體類型
這篇文章主要介紹了Python-docx 實現(xiàn)整體修改或者部分修改文字的大小和字體類型,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-03-03