11 基本排序算法:桶排序与计数排序

一、桶排序

原理

桶排序(Bucket sort)或所谓的箱排序,是一个排序算法,工作的原理是将数组分到有限数量的桶里。每个桶再个别排序(有可能再使用别的排序算法或是以递归方式继续使用桶排序进行排序),最后依次把各个桶中的记录列出来记得到有序序列。桶排序是鸽巢排序的一种归纳结果。当要被排序的数组内的数值是均匀分配的时候,桶排序使用线性时间(Θ(n))。但桶排序并不是比较排序,他不受到O(n log n)下限的影响。

基本思想

桶排序的思想近乎彻底的分治思想。
桶排序假设待排序的一组数均匀独立的分布在一个范围中,并将这一范围划分成几个子范围(桶)。
然后基于某种映射函数f ,将待排序列的关键字 k 映射到第i个桶中 (即桶数组B 的下标i) ,那么该关键字k 就作为 B[i]中的元素 (每个桶B[i]都是一组大小为N/M 的序列 )。
接着将各个桶中的数据有序的合并起来 : 对每个桶B[i] 中的所有元素进行比较排序 (可以使用快排)。然后依次枚举输出 B[0]….B[M] 中的全部内容即是一个有序序列。

补充: 映射函数一般是 f = array[i] / k; k^2 = n; n是所有元素个数

为了使桶排序更加高效,我们需要做到这两点:

  • 在额外空间充足的情况下,尽量增大桶的数量
  • 使用的映射函数能够将输入的 N 个数据均匀的分配到 K 个桶中

同时,对于桶中元素的排序,选择何种比较排序算法对于性能的影响至关重要。

实现逻辑
  • 设置一个定量的数组当作空桶子。
  • 寻访序列,并且把项目一个一个放到对应的桶子去。
  • 对每个不是空的桶子进行排序。
  • 从不是空的桶子里把项目再放回原来的序列中。
动画演示过程
9.桶排序.gif
Go语言描述

桶排序:通过开辟另外空间并分配+收集的排序方法

  • 平均时间复杂度:O(n*k)
  • 最坏时间复杂度:O(n²)
  • 最好时间复杂度:O(n)
  • 空间复杂度:O(n*k)
  • 稳定性:稳定
func BucketSort(list []int, size int) {

    min := list[0]
    max := list[0]

    for i := 0; i < len(list); i++ {
        if min > list[i] {
            min = list[i]
        }
        if max < list[i] {
            max = list[i]
        }
    }

    // 桶切片初始化
    count := make([][]int, (max-min)/size+1)

    // 数据入桶
    for i := 0; i < len(list); i++ {
        count[(list[i]-min)/size] = append(count[(list[i]-min)/size], list[i])
    }

    key := 0
    // 对每个桶进行排序
    for _, bucket := range count {
        if len(bucket) <= 0 {
            continue
        }
        // 插入排序
        InsertionSort(bucket)
        for _, value := range bucket {
            list[key] = value
            key = key + 1
        }
    }
    return
}

二、计数排序

计数排序不是比较数值排序,是记录数据出现次数的一种排序算法。它的原理有点类似桶排序算法,可以看似特殊的桶排序算法。

原理

算法解析:

  • 找出待排数组中最大值;
  • 额外一个数组记录待排数组值出现的次数;
  • 循环打印存储数值次数的数组下标值;
动画演示过程
计数排序.gif
Go语言描述
  • 计数排序 counting sort
  • 平均时间复杂度:O(n*k)
  • 最坏时间复杂度:O(n²)
  • 最好时间复杂度:O(n)
  • 空间复杂度:O(n*k)
  • 稳定性:稳定
package sort

func CountingSort(list []int) {
    // 初始化一个最大值,默认第一元素
    max := list[0]

    l := len(list)
    // 先找出最大值
    for i := 1; i < l; i++ {
        if max < list[i] {
            max = list[i]
        }
    }

    // 记录该值的出现次数
    arr := make([]int, max+1)
    for j := 0; j < l; j++ {
        arr[list[j]]++
    }
    // fmt.Println("arr:",arr)

    pos := 0
    for index, v := range arr {
        for x := 0; x < v; x++ {
            list[pos] = index
            pos++
        }
    }
}

由于计数排序依赖于额外数组保存整数元素出现的次数,需要开辟额外空间存储损耗较大(典型的空间换时间),不适合太过大量和离散的值,只适合小而紧凑的序列值的排序。

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