python查找第k小元素代碼分享
# -*- coding: utf-8 -*-
from random import randint
from math import ceil, floor
def _partition(A, l, r, i):
"""以A[i]為主元?jiǎng)澐謹(jǐn)?shù)組A[l..r],使得:
A[l..m-1] <= A[m] < A[m+1..r]
"""
A[i], A[r] = A[r], A[i] # i交換到末位r,作為主元
pivot = A[r] # 主元
m = l # 索引標(biāo)記
for n in xrange(l, r): # l..r-1
if A[n] <= pivot:
A[m], A[n] = A[n], A[m] # 交換
m += 1 # 后移
A[m], A[r] = A[r], A[m] # 主元到m位
return m
def _rand(A, l, r):
"""隨機(jī)劃分主元"""
return randint(l, r) # A[l..r]隨機(jī)取一個(gè)
def _select(A, l, r, k, pivot_selector = _rand):
"""利用快排,得A[l..r]中第k小的數(shù),k in [l+1,r+1]:
其尾遞歸方式,偽碼如下:
SELECT(A, l, r, k)
1 while true:
2 i ← ? // 劃分主元位置
3 m ← PARTITION(A, l, r, i) // 數(shù)組劃分
4 n ← m - l + 1 // A[l..m]元素個(gè)數(shù)
5 if k = n // 檢查A[m]是否是第k小的元素
6 then return A[m]
7 elseif k < n // 左劃分區(qū)
8 r = m - 1
9 else // 右劃分區(qū)
10 k = k - n
11 l = m + 1
Args:
pivot_selector(Function): 主元選取方法,默認(rèn)隨機(jī)方式
"""
if not A:
return None
if l == r:
return A[l]
while True:
i = pivot_selector(A, l, r)
m = _partition(A, l, r, i)
n = m - l + 1
if k == n:
return A[m]
elif k < n:
r = m - 1
else:
k = k - n
l = m + 1
def rand_select(A, k):
"""默認(rèn)隨機(jī)劃分主元方式,k in [1, len(A)]
E[T(n)] = O(n)
"""
return _select(A, 0, len(A) - 1, k);
def _median(A, l, r):
"""對(duì)A[l..r]插入排序(原地)后選取其中位數(shù)位置"""
for j in xrange(l, r + 1):
k = A[j]
i = j
while i > l and A[i-1] > k:
A[i] = A[i-1]
i -= 1
A[i] = k
return l + int((r - l) * 0.5) # 下中位數(shù)
def _medianOfMedians(A, l, r):
"""中位數(shù)的中位數(shù)方式:
1. 劃分為floor(n/5)個(gè)5元組,剩下(n%5)組成最后一組。
2. 找出ceil(n/5)個(gè)組各自的中位數(shù)。先對(duì)每組插入排序,再?gòu)闹羞x出中位數(shù)。
3. 對(duì)第2步中找出的ceil(n/5)個(gè)中位數(shù)重復(fù)上述操作,直到僅有一個(gè)中位數(shù)。
"""
if l == r:
return l
n = r - l + 1 # 元素個(gè)數(shù)
m = int(ceil(n / 5.0)) # 劃分組數(shù),每組5個(gè)元素
for i in xrange(m):
# 每組起始位和結(jié)束位
sub_l = l + i * 5
sub_r = sub_l + 4
if sub_r > r:
sub_r = r
# 對(duì)每組元素插入排序后,選取中位數(shù)
sub_m = _median(A, sub_l, sub_r) # 中位數(shù)索引
# 交換中位數(shù)到前幾位
j = l + i
A[j], A[sub_m] = A[sub_m], A[j]
return _medianOfMedians(A, l, l + m - 1) # 中位數(shù)的中位數(shù)
def bfprt_select(A, k):
"""中位數(shù)的中位數(shù)方式(BFPRT算法)
T(n) = O(n)
"""
return _select(A, 0, len(A) - 1, k, _medianOfMedians);
def _median3(A, l, r):
"""三數(shù)中位數(shù)方式,取l,r,(l+r)/2三數(shù)中位數(shù)"""
c = (l + r) / 2
keys = [l, c, r]
i = _median(keys, 0, 2)
return keys[i]
def median_select(A, k):
"""三數(shù)中位數(shù)方式,以消除最壞情況"""
return _select(A, 0, len(A) - 1, k, _median3);
if __name__ == '__main__':
import random, time
from copy import copy
print('preparing data...')
n = 1000000
nums = range(n)
random.shuffle(nums)
print('ready go!')
def timeit(fnc, *args, **kargs):
print('%s starts processing' % fnc.__name__)
begtime = time.clock()
retval = fnc(*args, **kargs)
endtime = time.clock()
print('%s takes time : %f' % (fnc.__name__, endtime - begtime))
return retval
test_methods = [rand_select, bfprt_select, median_select]
k = random.randrange(n) + 1
dashes = '---' * 10
for test in test_methods:
print(dashes)
nums_new = copy(nums)
result = timeit(test, nums_new, k)
print('the %dth smallest element: %d' % (k, result))
- Python cookbook(數(shù)據(jù)結(jié)構(gòu)與算法)保存最后N個(gè)元素的方法
- Python cookbook(數(shù)據(jù)結(jié)構(gòu)與算法)從任意長(zhǎng)度的可迭代對(duì)象中分解元素操作示例
- Python cookbook(數(shù)據(jù)結(jié)構(gòu)與算法)將序列分解為單獨(dú)變量的方法
- 利用Python找出序列中出現(xiàn)最多的元素示例代碼
- Python找出list中最常出現(xiàn)元素的方法
- Python實(shí)現(xiàn)的數(shù)據(jù)結(jié)構(gòu)與算法之鏈表詳解
- Python實(shí)現(xiàn)的數(shù)據(jù)結(jié)構(gòu)與算法之快速排序詳解
- Python實(shí)現(xiàn)的數(shù)據(jù)結(jié)構(gòu)與算法之隊(duì)列詳解
- Python實(shí)現(xiàn)的數(shù)據(jù)結(jié)構(gòu)與算法之基本搜索詳解
- Python數(shù)據(jù)結(jié)構(gòu)與算法之圖結(jié)構(gòu)(Graph)實(shí)例分析
- Python cookbook(數(shù)據(jù)結(jié)構(gòu)與算法)找到最大或最小的N個(gè)元素實(shí)現(xiàn)方法示例
相關(guān)文章
Python使用MD5加密算法對(duì)字符串進(jìn)行加密操作示例
這篇文章主要介紹了Python使用MD5加密算法對(duì)字符串進(jìn)行加密操作,結(jié)合實(shí)例形式分析了Python實(shí)現(xiàn)md5加密相關(guān)操作技巧,需要的朋友可以參考下2018-03-03
Python3內(nèi)置模塊之base64編解碼方法詳解
這篇文章主要介紹了Python3內(nèi)置模塊之base64編解碼方法詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-07-07
Python使用POP3和SMTP協(xié)議收發(fā)郵件的示例代碼
這篇文章主要介紹了Python使用POP3和SMTP協(xié)議收發(fā)郵件的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-04-04
Python在centos7.6上安裝python3.9的詳細(xì)教程(默認(rèn)python版本為2.7.5)
這篇文章主要介紹了Python在centos7.6上安裝python3.9(默認(rèn)python版本為2.7.5)的方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2020-10-10
Python 實(shí)現(xiàn)子類獲取父類的類成員方法
今天小編就為大家分享一篇Python 實(shí)現(xiàn)子類獲取父類的類成員方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-01-01
解決Python安裝cryptography報(bào)錯(cuò)問(wèn)題
這篇文章主要介紹了解決Python安裝cryptography報(bào)錯(cuò)問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-09-09
解決Python正則表達(dá)式匹配反斜杠''''\''''問(wèn)題
這篇文章主要介紹了Python正則表達(dá)式匹配反斜杠'\'問(wèn)題 ,很多朋友在使用python 正則式的過(guò)程中,經(jīng)常被這個(gè)問(wèn)題困擾,今天小編通過(guò)代碼給大家詳細(xì)介紹,需要的朋友可以參考下2019-07-07

