第二部分--排序和顺序统计学-第8章--线性时间排序

说明:该系列博客整理自《算法导论(原书第二版)》,但更偏重于实用,所以晦涩偏理论的内容未整理,请见谅。另外本人能力有限,如有问题,恳请指正!

1、计数排序

    计数排序 假设 n 个输入元素中的每一个都是介于0到 k 之间的整数,此处 k 为某个整数。当 k = O ( n )时,计数排序的时间复杂度为 Θ ( n )。

    计数排序的基本思想就是对每一个输入元素 x ,确定出小于 x 的元素个数。有了这一信息,就可以把 x 直接放到它在最终输出数组中的位置上。

    在计数排序算法的代码中,我们假定输入为数组A[1...n],length[A]=n。另外还需要两个数组:存放排序结果的B[1...n],以及提供临时存储区的C[0...k]。

    时间复杂度为时间复杂度为O(n+k),因此待排序的关键字范围0~k不宜过大。

COUNTING-SORT(A, B, k)

1      let C[0 .. k] be a new array

2      for i = 0 to k

3          C[i] = 0

4      for j = 1 to A.length

5          C[A[j]] = C[A[j]] + 1//在C[A[j]] 位置记录A[j]这个数出现的次数

6      // C[i] now contains the number of elements equal to i.

7      for i = 1 to k

8          C[i] = C[i] + C[i - 1]

9      // C[i] now contains the number of elements less than or equal to i.

10     for j = A.length downto 1

11        B[C[A[j]]] = A[j]

12        C[A[j]] = C[A[j]] - 1

    在1-2行中的for循环初始化操作之后;在3-4行要检查每个输入元素:如果输入元素值为i,则增加C[i]的值,于是经过3-4行后,C[i]中存放了等于i的元素的个数;在6-7行中,通过在数组C中记录计数和,可以确定对每一个i=0,1...,k,有多少输入元素是小于等于i的;在9-11行,将每个元素A[j]放到输出数组B中与其相对应的最终位置上。==>计数排序的使用前提是知道输入数组中的最大输入值k,通过大小为k的数组来统计输入数组中元素的个数及位置,最终通过对计数的统计得到最终的排序数组。

    计数排序的一个重要性质是它是稳定的:具有相同值得元素在输出数组中的相对次序与它们在输入数组中的次序相同。而且,计数排序经常作为基数排序算法的一个子程序

2、基数排序

    基数排序 要求待排序的元素拥有相同的位数,且从元素的低位到高位依次排序。基数排序算法的代码是很直观的,在下面伪码过程中,假设数组长度为n,每个元素都有d位数字,其中第1位为最低位,第d位是最高位。

    时间复杂度:设在基数排序中,r为基数,d为位数。则基数排序的时间复杂度为O(d(n+r))

RADIX-SORT(A, d)

1     for i = 1 to d

2         use a stable sort to sort array A on digit i

基数排序的简单Java实现

public static void radixSort(int[] array) { 

    int len= String.valueOf(array[0]).toString().length();

    for(int i= 0; i < len; i++)        

        radixQuickSort(array, i);

}

/** * 计数排序的变形 */

public static void radixQuickSort(int[] array, int power) {

    int[] result = new int[array.length];

    int[] temp=new int[10];

    for(int i= 0; i < 10; i++)       

         temp[i] = 0;

    for(int j= 0; j < array.length; j++)        

        temp[getDigit(array[j], power)]++;

    for(int i= 1; i < 10; i++)        

        temp[i] = temp[i] + temp[i - 1];

    for(int j= array.length - 1; j >= 0; j--) {        

        result[temp[getDigit(array[j], power)] - 1] = array[j];        

        temp[getDigit(array[j], power)]--;    

    }

    for(int i= 0; i < result.length; i++)        

        array[i] = result[i];}

/** * 获得数字第n位的值 */

public static int getDigit(int value, int power) {

    return (int) (value / Math.pow(10, power)) % 10;

}

3、桶排序

    当 桶排序 (bucket sort)的输入符合均匀分布时,它可以以线性时间运行。与计数排序类似,桶排序也对输入做了某种假设,因而运行的很快。具体来说,计数排序假设输入是由一个小范围内的整数构成,而桶排序假设输入由一个随机过程产生,该过程将元素均匀地分布在区间[ 0, 1 )上。

    桶排序的思想就是把区间[ 0, 1 )划分成 m 个相同大小的子区间,或称  。然后,将 n 个输入数分布到各个桶中去。因为输入数均匀分布在[ 0, 1 )上,所以,一般不会有很多数落在一个桶中的情况。为得到结果,先对各个桶中的元素进行排序,然后按次序把桶中的元素列出来即可。

    在桶排序算法的代码中,假设输入的是一个含有n个元素的数组A,且每个元素满足0<=A[i]<1。另外还需要一个辅助数组B[0...m-1]来存放(指向)链表(桶),并假设可以用某种机制来维护这些表。

    缺点:桶排序前提是知道排序的最大的数是几,然后就有多少个桶,然后将待排序的数依次放入桶中,但是这也是桶排序的缺点,即空间复杂度由最大的数决定。

    时间复杂度为O(m+n),m为桶的个数,n为数组元素个数。

BUCKET-SORT(A)

1  let B[0 .. m - 1] be a new array//创建B数组,每一个数组元素代表一个桶,上面挂一个链表

2  n = A.length

3  for i = 0 to m - 1

4          make B[i] an empty list//B数组每一个数组元素上面挂一个链表

5  for i = 1 to n

6          insert A[i] into list B[FLOOR(mA[i])]//FLOOR(mA[i])公式成立的前提是0<=A[i]<1,如果0<=A[i]<k,则公式为FLOOR(mA[i]/k)===这句代码的意思是将元素A[i]放入B的FLOOR(mA[i])这个元素上挂接的桶中

7  for i = 0 to m - 1//将所有的链表排序

8          sort list B[i] with insertion sort

9  concatenate the lists B[0], B[1], ..., B[n - 1] together in order

    如果输入数组均匀分布,则桶排序以线性期望时间运行。即使输入不符合均匀分布,桶排序也仍然可以以线性时间运行,只要输入满足这样一个性质,即各个桶尺寸的平方和与总的元素数呈线性关系。

4、参考

    1)、算法导论读书笔记(8)

    2)、排序算法之计数排序:该博客对计数排序做了改进,使得相对来说实现了原地排序。

    3)、计数排序c语言实现:与博客中伪码对应的c语言实现,没有任何改进,无亮点。

    4)、基数排序:与博客中伪码对应的c语言实现,没有任何改进,无亮点。

    5)、桶排序:这个算法对桶排序时间复杂的的计算做了说明,我自己懒得写,记录。

    6)、桶排序算法:讲解的通俗易懂。

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

推荐阅读更多精彩内容