算法专题:Linear Sort

Linear Sort即线性排序,指的是一系列能做到线性时间复杂度即O(n)的排序算法,这里主要介绍三个:桶排序(bucket sort),计数排序(count sort)和基数排序(radix sort)。
排序算法基于两类,一类是基于比较的排序,常规排序一般就是这类,例如快速排序、归并排序、堆排序。这种排序方法有着O(nlgn)的下限限制(已有证明比较排序不可能做到比O(nlgn)好)。而非比较排序没有这个限制。虽然这些排序方法看上去复杂度比常规的的时间复杂度O(nlgn)要好,其实都有一些其他方面的限制,所以还是要看情况进行使用。

  1. 计数排序
    顾名思义,就是通过计算各个元素出现的次数来排序。 假设出现的元素都在0-k之间,将这k+1个数进行频次计数,然后再从小到大把这些数给排出来。代码如下:
def countingsort( aList, k ):
    counter = [0] * ( k + 1 )
    for i in aList:
        counter[i] += 1

    ndx = 0
    for i in range( len( counter ) ):
        while 0 < counter[i]:
            aList[ndx] = i
            ndx += 1
            counter[i] -= 1 
整个算法可以分成两块,第一块计数,时间复杂度O(n),第二块摆放,时间复杂度O(k),因此总的时间复杂度是O(n+k),空间复杂度也是O(n+k)。很明显,适用的情况,仅限于所需排序的序列是0-k的整数而且k比较小。
  1. 桶排序
    桶排序自然要用到“桶”,所谓的“桶”就是一个个容器,其能容纳一个区间内未排序的数值。利用“桶”,将数值按照区域分好,然后各自排序,再连到一起,就是桶排序的思想。桶内排序,可以使用比较排序算法。

    def bucketsort(A):
        # get hash codes
        code = hashing(A)
        buckets = [list() for _ in range(code[1])]
        # distribute data into buckets: O(n)
        for i in A:
            x = re_hashing(i, code)
            buck = buckets[x]
            buck.append(i)
    
        for bucket in buckets:
            bucket.sort()
    
        ndx = 0
        # merge the buckets: O(n)
        for b in range(len(buckets)):
            for v in buckets[b]:
                A[ndx] = v
                ndx += 1
    
    import math
    def hashing(A):
        m = A[0]
        for i in range(1, len(A)):
            if (m < A[i]):
            m = A[i]
        result = [m, int(math.sqrt(len(A)))]
        return result
    def re_hashing(i, code):
        return int(i / code[0] * (code[1] - 1)) 
    

首先是把数据压缩到01的区间内,这里的代码是取n^(1/2)个桶。假设数组内的最大值是K,有M个桶,那么相当于把0-K分成M个区间,即0M-1号桶。对于某个数x,应该分到序号为(M-1)*x/K,因为x=0时分到0号桶,x=K时分到第M-1号桶。
然后分完数之后对桶内进行排序,排序完之后连起来。
稍微分析一下复杂度。最坏情况就是所有数据都在一个桶里面,那么相当于白干,还不如直接常规排序。
最好情况就是数据平均分布,总共N个元素M个桶,每个桶就是N/M个元素,总时间复杂度就是O(N+Nlog(N/M)),当M越大时总复杂度越小,但同时空间复杂度也就更高。
此外,只要桶排序内部排序算法是稳定的,那么整个桶排序也就是稳定的。

  1. 基数排序
    基数排序属于多key排序,一般是从最小的key分堆排序,然后再对较大的key排序,直至对所有key完成排序。
    网上很常见的一个例子就是对十进制整数排序,因为每一位的权重不一样,因此也是对key分次排序的一个例子。
    例如170, 45, 75, 90, 802, 2, 24, 66,排序过程如下:
    先对个位进行排序:170,90,802,2,24,45,75,66
    然后进行十位排序:802,2,24,45,66,170,75,90
    最后对百位排序:2,24,45,66,75,90,175,802
    当然,key不止限定于十进制的权重。不过基本上就是这么一个思路。假如有k个key,那么一共要排k次,如果使用桶排序,那么就是O(nk)。
def radixSort(a, n, maxLen):
    for x in range(maxLen):
        bins = [[] for i in range(n)]
        for y in a:
            bins[(y // 10 ** x) % n].append(y)
        a = []
        for section in bins:
            a.extend(section)
      return a 

上面的代码,如果是对十进制排序,那至少要有10个桶来放,n=10.maxLen是指数值的最大长度用来确定排序次数。

总结:线性排序总体来说限制比较多,不够灵活,但是在特定场合还是可以用。

例题:maxGap。给出一个未排序的非负整数数组,找出其排序状况下的相邻两个数的最大差值。要求线性时间。
【解】这个题就是线性排序的用武之地。假如不排序,几乎没法做。当然也可以用Quick Sort强行线性(其复杂度从O(n)~O(n^2)),不过用正宗的线性排序显然更好。
对这道题而言,计数排序明显不靠谱,基数排序并没有多key,所以桶排序是最佳的选择。事实上,桶排序时可以不用在桶里面sort,因为题目只是要求max gap,因此如果能知道每一个桶内的max和min,那么桶与桶之间的gap就知道了。至于桶内的呢?平均gap是(max-min)/(N-1),如果让每个桶刚好覆盖平均gap大小,那么极端情况就是每个桶一个元素,桶内gap不用考虑;否则,某些桶多某些桶少,出现分布不均匀的情况时,最大gap也只可能出现在桶与桶之间。

# Time:  O(n)
# Space: O(n)
class Solution:
     # @param numss: a list of integers
     # @return: the maximum difference
    def maximumGap(self, nums):
        if len(nums) < 2:
            return 0
        
        # Init bucket.
        max_val, min_val = max(nums), min(nums)
        gap = max(1, (max_val - min_val) // (len(nums) - 1))
        bucket_size = (max_val - min_val) // gap + 1
        bucket = [{'min': float("inf"), 'max': float("-inf")} for _ in range(bucket_size)]

        # Find the bucket where the n should be put.
        for n in nums:
            # min_val / max_val is in the first / last bucket.
            if n in (max_val, min_val):
                continue      
            i = (n - min_val) // gap
            bucket[i]['min'] = min(bucket[i]['min'], n)
            bucket[i]['max'] = max(bucket[i]['max'], n)
        
        # Count each bucket gap between the first and the last bucket.
        max_gap, pre_bucket_max = 0, min_val
        for i in range(bucket_size):
            # Skip the bucket it empty.
            if bucket[i]['min'] == float("inf") and bucket[i]['max'] == float("-inf"):
                continue
            max_gap = max(max_gap, bucket[i]['min'] - pre_bucket_max)
            pre_bucket_max = bucket[i]['max']
        # Count the last bucket.
        max_gap = max(max_gap, max_val - pre_bucket_max) 
        
        return max_gap 
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,294评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,493评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,790评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,595评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,718评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,906评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,053评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,797评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,250评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,570评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,711评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,388评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,018评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,796评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,023评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,461评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,595评论 2 350

推荐阅读更多精彩内容

  • 背景 一年多以前我在知乎上答了有关LeetCode的问题, 分享了一些自己做题目的经验。 张土汪:刷leetcod...
    土汪阅读 12,740评论 0 33
  • 提高沟通效果的三大要素即本能、情感 和逻辑 ,本能就是一个人最直白的简单的印象,要想提高沟通效果,就需要给人一种可...
    爱学习的小胡同学阅读 184评论 0 0
  • 微博真的是一个好东西,至少K小姐是这么认为。访问对方的主页不会留下痕迹,有时候为了了解一个人甚至可以一口气扒完对方...
    李布波阅读 500评论 2 6
  • 古都长安旗猎猎,渭水河畔书朗朗。 谈笑鸿儒表壮志,往来骄子续新章。 不畏军营苦与累,自古红炉出好钢。 秦皇汉武俱往...
    垚君阅读 167评论 0 0