# 基数排序 上节介绍的计数排序适用于数据量 $n$ 大但数据范围 $m$ 不大的情况。假设需要排序 $n = 10^6$ 个学号数据,学号是 $8$ 位数字,那么数据范围 $m = 10^8$ 很大,使用计数排序则需要开辟巨大的内存空间,而基数排序则可以避免这种情况。 「基数排序 Radix Sort」主体思路与计数排序一致,也通过统计出现次数实现排序,**并在此基础上利用位与位之间的递进关系,依次对每一位执行排序**,从而获得排序结果。 ## 算法流程 以上述的学号数据为例,设数字最低位为第 $1$ 位、最高位为第 $8$ 位,基数排序的流程为: 1. 初始化位数 $k = 1$ ; 2. 对学号的第 $k$ 位执行「计数排序」,完成后,数据即按照第 $k$ 位从小到大排序; 3. 将 $k$ 自增 $1$ ,并返回第 `2.` 步继续迭代,直至排序完所有位后结束; ![基数排序算法流程](radix_sort.assets/radix_sort_overview.png) 下面来剖析代码实现。对于一个 $d$ 进制的数字 $x$ ,其第 $k$ 位 $x_k$ 的计算公式为 $$ x_k = \lfloor\frac{x}{d^{k-1}}\rfloor \mod d $$ 其中 $\lfloor a \rfloor$ 代表对浮点数 $a$ 执行向下取整,$\mod d$ 代表对 $d$ 取余。学号数据的 $d = 10$ , $k \in [1, 8]$ 。 此外,我们需要小幅改动计数排序代码,使之可以根据数字第 $k$ 位执行排序。 === "Java" ```java title="radix_sort.java" [class]{radix_sort}-[func]{digit} [class]{radix_sort}-[func]{countingSortDigit} [class]{radix_sort}-[func]{radixSort} ``` === "C++" ```cpp title="radix_sort.cpp" [class]{}-[func]{digit} [class]{}-[func]{countingSortDigit} [class]{}-[func]{radixSort} ``` === "Python" ```python title="radix_sort.py" [class]{}-[func]{digit} [class]{}-[func]{counting_sort_digit} [class]{}-[func]{radix_sort} ``` === "Go" ```go title="radix_sort.go" [class]{}-[func]{digit} [class]{}-[func]{countingSortDigit} [class]{}-[func]{radixSort} ``` === "JavaScript" ```javascript title="radix_sort.js" [class]{}-[func]{digit} [class]{}-[func]{countingSortDigit} [class]{}-[func]{radixSort} ``` === "TypeScript" ```typescript title="radix_sort.ts" [class]{}-[func]{digit} [class]{}-[func]{countingSortDigit} [class]{}-[func]{radixSort} ``` === "C" ```c title="radix_sort.c" [class]{}-[func]{digit} [class]{}-[func]{countingSortDigit} [class]{}-[func]{radixSort} ``` === "C#" ```csharp title="radix_sort.cs" [class]{radix_sort}-[func]{digit} [class]{radix_sort}-[func]{countingSortDigit} [class]{radix_sort}-[func]{radixSort} ``` === "Swift" ```swift title="radix_sort.swift" [class]{}-[func]{digit} [class]{}-[func]{countingSortDigit} [class]{}-[func]{radixSort} ``` === "Zig" ```zig title="radix_sort.zig" [class]{}-[func]{digit} [class]{}-[func]{countingSortDigit} [class]{}-[func]{radixSort} ``` !!! question "为什么从最低位开始排序?" 对于先后两轮排序,第二轮排序可能会覆盖第一轮排序的结果,比如第一轮认为 $a < b$ ,而第二轮认为 $a > b$ ,则第二轮会取代第一轮的结果。由于数字高位比低位的优先级更高,所以要先排序低位再排序高位。 ## 算法特性 **时间复杂度 $O(n k)$** :设数据量为 $n$ 、数据为 $d$ 进制、最大为 $k$ 位,则对某一位执行计数排序使用 $O(n + d)$ 时间,排序 $k$ 位使用 $O((n + d)k)$ 时间;一般情况下 $d$ 和 $k$ 都比较小,此时时间复杂度近似为 $O(n)$ 。 **空间复杂度 $O(n + d)$** :与计数排序一样,借助了长度分别为 $n$ , $d$ 的数组 `res` 和 `counter` ,因此是“非原地排序”。 与计数排序一致,基数排序也是稳定排序。相比于计数排序,基数排序可适用于数值范围较大的情况,**但前提是数据必须可以被表示为固定位数的格式,且位数不能太大**。比如浮点数就不适合使用基数排序,因为其位数 $k$ 太大,可能时间复杂度 $O(nk) \gg O(n^2)$ 。