python查找第k小元素代码分享 -电脑资料

电脑资料 时间:2019-01-01 我要投稿
【www.unjs.com - 电脑资料】

   

    复制代码代码如下:

    # -*- coding: utf-8 -*-

    from random import randint

    from math import ceil, floor

    def _partition(A, l, r, i):

    """以A[i]为主元划分数组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 # 索引标记

    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):

    """随机划分主元"""

    return randint(l, r) # A[l..r]随机取一个

    def _select(A, l, r, k, pivot_selector = _rand):

    """利用快排,得A[l..r]中第k小的数,k in [l+1,r+1]:

    其尾递归方式,伪码如下:

    SELECT(A, l, r, k)

    1 while true:

    2   i ← ? // 划分主元位置

    3   m ← PARTITION(A, l, r, i) // 数组划分

    4   n ← m - l + 1 // A[l..m]元素个数

    5   if k = n // 检查A[m]是否是第k小的元素

    6     then return A[m]

    7   elseif k < n // 左划分区

    8     r = m - 1

    9   else // 右划分区

    10    k = k - n

    11    l = m + 1

    Args:

    pivot_selector(Function): 主元选取方法,默认随机方式

    """

    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):

    """默认随机划分主元方式,k in [1, len(A)]

    E[T(n)] = O(n)

    """

    return _select(A, 0, len(A) - 1, k);

    def _median(A, l, r):

    """对A[l..r]插入排序(原地)后选取其中位数位置"""

    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) # 下中位数

    def _medianOfMedians(A, l, r):

    """中位数的中位数方式:

    1. 划分为floor(n/5)个5元组,剩下(n%5)组成最后一组,

python查找第k小元素代码分享

    2. 找出ceil(n/5)个组各自的中位数。先对每组插入排序,再从中选出中位数。

    3. 对第2步中找出的ceil(n/5)个中位数重复上述操作,直到仅有一个中位数。

    """

    if l == r:

    return l

    n = r - l + 1 # 元素个数

    m = int(ceil(n / 5.0)) # 划分组数,每组5个元素

    for i in xrange(m):

    # 每组起始位和结束位

    sub_l = l + i * 5

    sub_r = sub_l + 4

    if sub_r > r:

    sub_r = r

    # 对每组元素插入排序后,选取中位数

    sub_m = _median(A, sub_l, sub_r) # 中位数索引

    # 交换中位数到前几位

    j = l + i

    A[j], A[sub_m] = A[sub_m], A[j]

    return _medianOfMedians(A, l, l + m - 1) # 中位数的中位数

    def bfprt_select(A, k):

    """中位数的中位数方式(BFPRT算法)

    T(n) = O(n)

    """

    return _select(A, 0, len(A) - 1, k, _medianOfMedians);

    def _median3(A, l, r):

    """三数中位数方式,取l,r,(l+r)/2三数中位数"""

    c = (l + r) / 2

    keys = [l, c, r]

    i = _median(keys, 0, 2)

    return keys[i]

    def median_select(A, k):

    """三数中位数方式,以消除最坏情况"""

    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))

    QQ空间 搜狐微博 人人网 开心网 百度搜藏更多

    Tags:python 第k小元素

    复制链接收藏本文打印本文关闭本文返回首页

    上一篇:python获取beautifulphoto随机某图片代码实例

    下一篇:windows下wxPython开发环境安装与配置方法

   

相关文章

2014-02-02python求斐波那契数列示例分享

2012-10-10python笔记(1) 关于我们应不应该继续学习python

2009-03-03Python字符串的encode与decode研究心得乱码问题解决方法

2014-02-02Python的print用法示例

2012-08-08Python运行的17个时新手常见错误小结

2014-02-02使用python在校内发人人网状态(人人网看状态)

2014-05-05使用python实现拉钩网上的FizzBuzzWhizz问题示例

2013-11-11python解析json实例方法

2014-01-01python使用 api上传图片到微博示例

2006-10-10学习python (2)

   

文章评论

   

最 近 更 新

   

Python 第一步 hello world

一篇不错的Python入门教程

python根据经纬度计算距离示例

Python中使用中文的方法

Python操作Mysql实例代码教程在线版(查询

python的id()函数介绍

浅析python 内置字符串处理函数的使用方法

python3使用tkinter实现ui界面简单实例

python的几种开发工具介绍

Python 调用DLL操作抄表机

   

热 点 排 行

   

Python入门教程 超详细1小时学会

python 中文乱码问题深入分析

比较详细Python正则表达式操作指

Python字符串的encode与decode研

Python open读写文件实现脚本

Python enumerate遍历数组示例应

Python 深入理解yield

Python+Django在windows下的开发

python 文件和路径操作函数小结

python 字符串split的用法分享

最新文章