基数排序简析

基数排序是一种适用于特定数据类型的内部排序算法。这种排序算法要求数据必须能够划分为多个排序关键字,且这些排序关键字应该有优先级的区别。比如某个序列的数据都是整数,且取值范围在[0,99],则我们可以划分出两个关键字:十位数和个位数。其中十位数比个位数的优先级要高。

一种基数排序的方法,是从优先级最高的关键字开始,将序列中各值分配为多个子序列,然后对每个子序列进行排序(这一次排序可以采用次高优先级关键字继续基数排序,也可以采用其他排序方法),然后将各个子序列顺序连接——因为最高优先级关键字是排好序的,各子序列必然也是有序连接的。这种又称为MSD法。

这种方法很明显和快速排序一样,都是分治策略。

另一种基数排序的方法,是从优先级最低的关键字开始,将序列中各值分配为多个子序列,按关键字顺序连接各子序列之后,再以优先级第二低的关键字进行第二次分配-连接……如此往复直到最高优先级的关键字也进行了分配连接过程,排序完毕。这种又称为LSD法。这种排序的要求是,每一次分配的时候,应保存序列的稳定性——也就是说,如果两个数据被分配到同一个子序列,则其先后顺序应保持稳定——这就代表着,我们对更高一级关键字进行分配的时候,同一子序列中的低级关键字产生的顺序不会被破坏。

可见,采用LSD法时,时间复杂度为O(d * n),其中d为关键字数目,一般稳定,所以时间复杂度为O(n)。MSD法,如果各层都采用基数排序,同样是O(d * n),但是会有额外的内存花费;如果下面几层采用别的算法,就比较复杂了,且不提。

通常我们说基数排序,默认指的就是LSD法。MSD法往往用于类似先区分男女再按身高排序之类的情况。

LSD法速度快,缺点一是对数据有类型要求,二是在顺序存储结构(如数组)时很难进行分配。典型用法是对线性链表进行排序。

仍然以[0,99]范围内的数组进行排序为例:

1)将数组转化为单链表A。

2)我们另外制作出一个链表B,该链表有10个节点依次表示0-9。

3)另设10个指针分别指向这10个节点,节点和指针之间的链表就是根据关键字分配来的序列。

4)将单链表A的元素依次取出插入链表B:提取关键字,然后分配到对应的指针那里,插入指针后面然后将指针后移。

(可见节点始终指向各分配序列的尾部,而指针始终指向各分配序列的头部)

5)将节点取出来,连接空隙,就可以得到经过一次排序的链表了。

6)对更高一级优先级的关键字进行同样的操作,直到最高优先级关键字。

7)将链表的值转交到数组,销毁申请的内存,完毕。

以下是golang编写的算法代码:


package main

import (
    "fmt"
    "math/rand"
    "time"
)

// 基数排序
func RadixSort(sz []int) {
    // 链表的数据类型
    type ele struct {
        data int
        next *ele
    }

    // 将切片转化为链表
    n := len(sz)
    r := make([]ele, n+10) // 保管要排序的数据
    t := make([]*ele, 10)  // 记录各分段的段尾
    s := r[n:]             // 作为各分段的段头
    p := &r[0]             // 记录链表的头部
    for i := 0; i < n; i++ {
        r[i].data = sz[i]
        r[i].next = &r[i+1]
    }
    r[n-1].next = nil

    // 预制分成十段的链表,每段都有头尾(目前二者相同)
    initialize := func() {
        for i := 0; i < 9; i++ {
            t[i] = &s[i]
            s[i].next = &s[i+1]
        }
        t[9], s[9].next = &s[9], nil
    }

    // 将段中的预置的段头剔除出来,必须从后向前(请思考)
    separate := func() *ele {
        for i := 9; i > 0; i-- {
            t[i-1].next = s[i].next
        }
        return s[0].next
    }

    // 依次取出链表中元素顺序插入各段中
    insert := func(fac func(int) int) {
        for p != nil {
            q := p.next
            j := fac(p.data)
            p.next = t[j].next
            t[j].next = p
            t[j], p = p, q
        }
    }

    // 按照个位排序第一次
    initialize()
    insert(func(k int) int { return k % 10 })
    p = separate()

    // 按照十位排序第二次
    initialize()
    insert(func(k int) int { return k / 10 % 10 })
    p = separate()

    // 按照百位排序第三次
    initialize()
    insert(func(k int) int { return k / 100 % 10 })
    p = separate()

    // 从链表返回给数组
    for i := 0; p != nil; i, p = i+1, p.next {
        sz[i] = p.data
    }
}

func main() {
    rand.Seed(time.Now().UnixNano())
    sz := make([]int, 30)
    for i := 0; i < 30; i++ {
        sz[i] = rand.Intn(90) + 10
    }
    fmt.Println(sz)
    RadixSort(sz)
    fmt.Println(sz)
}
时间: 2024-10-11 10:33:47

基数排序简析的相关文章

web应用构架LAMT及tomcat负载简析

Httpd    (mod_jk.so) workers.properties文件 uriworkermap.properties文件 <--AJP1.3--> Tomcat  --> jdk 大致流程:apache服务器通过mod_jk.so 模块处理jsp文件的动态请求.通过tomcat worker等待执行servlet/JSP的tomcat实例.使用 AJP1.3协议与tomcat通信.tomcat有借助jdk解析. 负载就是 多台tomcat.共同解析apache发送的jsp请

CentOS的网络配置简析

我们在进行对CentOS的网络配置时,一般会从IP地址(IPADDR).子网掩码(NETMASK).网关(Gateway).主机名(HOSTNAME).DNS服务器等方面入手.而在CentOS中,又有着不同的命令或配置文件可以完成这些配置操作,接下来,我们将从ifcfg系命令,iproute2系命令以及配置文件3个方面来简析网络配置的方法. 一.ifcfg系命令 ifcfg系命令包括ifconfig,route,netstat和hostname. 1.ifconfig命令 用来配置一个网络接口.

JDK源码简析--java.lang包中的基础类库

题记 JDK,Java Development Kit. 我们必须先认识到,JDK只是,仅仅是一套Java基础类库而已,是Sun公司开发的基础类库,仅此而已,JDK本身和我们自行书写总结的类库,从技术含量来说,还是在一个层级上,它们都是需要被编译成字节码,在JRE中运行的,JDK编译后的结果就是jre/lib下得rt.jar,我们学习使用它的目的是加深对Java的理解,提高我们的Java编码水平. 本系列所有文章基于的JDK版本都是1.7.16. 本节内容 在本节中,简析java.lang包所包

经验模态分解法简析 (转)

http://blog.sina.com.cn/s/blog_55954cfb0102e9y2.html 美国工程院士黄锷博士于1998年提出的一种信号分析方法:重点是黄博士的具有创新性的经验模态分解(Empirical Mode Decomposition)即EMD法,它是一种自适应的数据处理或挖掘方法,非常适合非线性,非平稳时间序列的处理,本质上是对数据序列或信号的平稳化处理. 1:关于时间序列平稳性的一般理解: 所谓时间序列的平稳性,一般指宽平稳,即时间序列的均值和方差为与时间无关的常数,

Java Annotation 及几个常用开源项目注解原理简析

PDF 版: Java Annotation.pdf, PPT 版:Java Annotation.pptx, Keynote 版:Java Annotation.key 一.Annotation 示例 Override Annotation Java 1 2 3 @Override public void onCreate(Bundle savedInstanceState); Retrofit Annotation Java 1 2 3 @GET("/users/{username}&quo

Linux网络性能优化方法简析

Linux网络性能优化方法简析 2010-12-20 10:56 赵军 IBMDW 字号:T | T 性能问题永远是永恒的主题之一,而Linux在网络性能方面的优势则显而易见,这篇文章是对于Linux内核中提升网络性能的一些优化方法的简析,以让我们去后台看看魔术师表演用的盒子,同时也看看内核极客们是怎样灵活的,渐进的去解决这些实际的问题. AD:2014WOT全球软件技术峰会北京站 课程视频发布 对于网络的行为,可以简单划分为 3 条路径:1) 发送路径,2) 转发路径,3) 接收路径,而网络性

.NET设计模式简析

首先,是设计模式的分类,我们知道,常用的设计模式共23种.但总体来说,设计模式氛围三大类: 创建型模式,共五种:工厂方法模式.抽象工厂模式.单列模式.建造者模式.原型模式. 结构型模式,共七种:适配器模式.装饰器模式.代理模式.外观模式.桥接模式.组合模式.享元模式. 行为型模式,共十一种:策略模式.模版方法模式.观察者模式.迭代子模式.责任链模式.命令模式.备忘录模式.转改模式.访问者模式.终结者模式.解释器模式. 另外还有并发型模式和线程池模式等. 介绍了分类,下面简单说下设计模式的六大原则

SpringMVC学习——概念、流程图、源码简析(一)

学习资料:开涛的<跟我学SpringMVC.pdf> 众所周知,springMVC是比较常用的web框架,通常整合spring使用.这里抛开spring,单纯的对springMVC做一下总结. 概念 HandlerMapping:处理器映射,对请求的URL进行映射为具体的处理器(如果有拦截器也包含拦截器,会将Handler和多个HandlerInterceptor封装为HandlerExecutionChain对象) HandlerAdapter:处理器适配器,适配不同类型的处理器,如Cont

Android WebView远程代码执行漏洞简析

0x00 本文参考Android WebView 远程代码执行漏洞简析.代码地址为,https://github.com/jltxgcy/AppVulnerability/tree/master/WebViewFileDemo.下面我们分析代码. 0x01 首先列出项目工程目录: MainActivity.java的代码如下: public class MainActivity extends Activity { private WebView webView; private Uri mUr