README.zh-CN.md 18.2 KB
Newer Older
琚致远 已提交
1 2
# JavaScript 算法与数据结构

3
[![CI](https://github.com/trekhleb/javascript-algorithms/workflows/CI/badge.svg)](https://github.com/trekhleb/javascript-algorithms/actions?query=workflow%3ACI+branch%3Amaster)
琚致远 已提交
4 5 6 7
[![codecov](https://codecov.io/gh/trekhleb/javascript-algorithms/branch/master/graph/badge.svg)](https://codecov.io/gh/trekhleb/javascript-algorithms)

本仓库包含了多种基于 JavaScript 的算法与数据结构。

H
Henry 已提交
8
每种算法和数据结构都有自己的 README,包含相关说明和链接,以便进一步阅读 (还有 YouTube 视频) 。
琚致远 已提交
9

O
Oleksii Trekhleb 已提交
10 11
_Read this in other languages:_
[_English_](https://github.com/trekhleb/javascript-algorithms/),
O
Oleksii Trekhleb 已提交
12 13
[_繁體中文_](README.zh-TW.md),
[_한국어_](README.ko-KR.md),
M
Minki-Kim 已提交
14
[_日本語_](README.ja-JP.md),
O
Oleksii Trekhleb 已提交
15
[_Polski_](README.pl-PL.md),
O
Oleksii Trekhleb 已提交
16
[_Français_](README.fr-FR.md),
O
Oleksii Trekhleb 已提交
17
[_Español_](README.es-ES.md),
O
Oleksii Trekhleb 已提交
18
[_Português_](README.pt-BR.md),
O
Oleksii Trekhleb 已提交
19
[_Русский_](README.ru-RU.md),
O
Oleksii Trekhleb 已提交
20
[_Türk_](README.tr-TR.md),
O
Oleksii Trekhleb 已提交
21
[_Italiana_](README.it-IT.md),
22
[_Bahasa Indonesia_](README.id-ID.md),
O
Oleksii Trekhleb 已提交
23 24
[_Українська_](README.uk-UA.md),
[_Arabic_](README.ar-AR.md)
O
Oleksii Trekhleb 已提交
25

H
Henry 已提交
26 27
*注意:这个项目仅用于学习和研究,**不是**用于生产环境。*

琚致远 已提交
28 29
## 数据结构

H
Henry 已提交
30
数据结构是在计算机中组织和存储数据的一种特殊方式,使得数据可以高效地被访问和修改。更确切地说,数据结构是数据值的集合,表示数据之间的关系,也包括了作用在数据上的函数或操作。
31

H
Henry 已提交
32
`B` - 初学者, `A` - 进阶
33

34 35 36 37
* `B` [链表](src/data-structures/linked-list/README.zh-CN.md)
* `B` [双向链表](src/data-structures/doubly-linked-list/README.zh-CN.md)
* `B` [队列](src/data-structures/queue/README.zh-CN.md)
* `B` [](src/data-structures/stack/README.zh-CN.md)
38
* `B` [哈希表(散列)](src/data-structures/hash-table/README.zh-CN.md)
H
Henry 已提交
39
* `B` [](src/data-structures/heap/README.zh-CN.md) - 最大堆 & 最小堆
40
* `B` [优先队列](src/data-structures/priority-queue/README.zh-CN.md)
41
* `A` [字典树](src/data-structures/trie/README.zh-CN.md)
42
* `A` [](src/data-structures/tree/README.zh-CN.md)
O
Oleksii Trekhleb 已提交
43 44 45
  * `A` [二叉查找树](src/data-structures/tree/binary-search-tree)
  * `A` [AVL 树](src/data-structures/tree/avl-tree)
  * `A` [红黑树](src/data-structures/tree/red-black-tree)
H
Henry 已提交
46
  * `A` [线段树](src/data-structures/tree/segment-tree) - 使用 `最小/最大/总和` 范围查询示例
O
Oleksii Trekhleb 已提交
47
  * `A` [树状数组](src/data-structures/tree/fenwick-tree) (二叉索引树)
48
* `A` [](src/data-structures/graph/README.zh-CN.md) (有向图与无向图)
49
* `A` [并查集](src/data-structures/disjoint-set)
O
Oleksii Trekhleb 已提交
50
* `A` [布隆过滤器](src/data-structures/bloom-filter)
琚致远 已提交
51 52 53

## 算法

H
Henry 已提交
54 55 56
算法是如何解决一类问题的明确规范。算法是一组精确定义操作序列的规则。

`B` - 初学者, `A` - 进阶
琚致远 已提交
57 58 59

### 算法主题

H
Henry 已提交
60
* **数学**
61
  * `B` [位运算](src/algorithms/math/bits) - set/get/update/clear 位、乘以/除以二进制位 、变负等
62
  * `B` [阶乘](src/algorithms/math/factorial/README.zh-CN.md)
H
Henry 已提交
63
  * `B` [斐波那契数](src/algorithms/math/fibonacci) - `经典``闭式` 版本
64
  * `B` [素数检测](src/algorithms/math/primality-test) (排除法)
65
  * `B` [欧几里得算法](src/algorithms/math/euclidean-algorithm) - 计算最大公约数 (GCD)
66
  * `B` [最小公倍数](src/algorithms/math/least-common-multiple) (LCM)
H
Henry 已提交
67 68
  * `B` [素数筛](src/algorithms/math/sieve-of-eratosthenes) - 查找任意给定范围内的所有素数
  * `B` [判断 2 次方数](src/algorithms/math/is-power-of-two) - 检查数字是否为 2 的幂 (原生和按位算法)
O
Oleksii Trekhleb 已提交
69
  * `B` [杨辉三角形](src/algorithms/math/pascal-triangle)
H
Henry 已提交
70 71 72
  * `B` [复数](src/algorithms/math/complex-number) - 复数及其基本运算
  * `B` [弧度和角](src/algorithms/math/radian) - 弧度与角的相互转换
  * `B` [快速算次方](src/algorithms/math/fast-powering)
73
  * `A` [整数拆分](src/algorithms/math/integer-partition)
H
Henry 已提交
74 75
  * `A` [割圆术](src/algorithms/math/liu-hui) - 基于 N-gons 的近似 π 计算
  * `A` [离散傅里叶变换](src/algorithms/math/fourier-transform) - 把时间信号解析成构成它的频率
琚致远 已提交
76
* **集合**
77
  * `B` [笛卡尔积](src/algorithms/sets/cartesian-product) - 多集合结果
H
Henry 已提交
78
  * `A` [洗牌算法](src/algorithms/sets/fisher-yates) - 随机置换有限序列
79 80 81 82 83 84
  * `A` [幂集](src/algorithms/sets/power-set) - 该集合的所有子集
  * `A` [排列](src/algorithms/sets/permutations) (有/无重复)
  * `A` [组合](src/algorithms/sets/combinations) (有/无重复)
  * `A` [最长公共子序列](src/algorithms/sets/longest-common-subsequence) (LCS)
  * `A` [最长递增子序列](src/algorithms/sets/longest-increasing-subsequence)
  * `A` [最短公共父序列](src/algorithms/sets/shortest-common-supersequence) (SCS)
H
Henry 已提交
85 86
  * `A` [背包问题](src/algorithms/sets/knapsack-problem) - `0/1``无边界` 问题
  * `A` [最大子数列问题](src/algorithms/sets/maximum-subarray) - `BF 算法``动态规划`
87
  * `A` [组合求和](src/algorithms/sets/combination-sum) - 查找形成特定总和的所有组合
琚致远 已提交
88
* **字符串**
89
  * `B` [汉明距离](src/algorithms/string/hamming-distance) - 符号不同的位置数
H
Henry 已提交
90 91
  * `A` [莱温斯坦距离](src/algorithms/string/levenshtein-distance) - 两个序列之间的最小编辑距离
  * `A` [Knuth–Morris–Pratt 算法](src/algorithms/string/knuth-morris-pratt) KMP 算法 - 子串搜索 (模式匹配)
Y
Yong Yang 已提交
92
  * `A` [字符串快速查找](src/algorithms/string/z-algorithm) - 子串搜索 (模式匹配)
H
Henry 已提交
93
  * `A` [Rabin Karp 算法](src/algorithms/string/rabin-karp) - 子串搜索
94
  * `A` [最长公共子串](src/algorithms/string/longest-common-substring)
H
Henry 已提交
95
  * `A` [正则表达式匹配](src/algorithms/string/regular-expression-matching)
琚致远 已提交
96
* **搜索**
O
Oleksii Trekhleb 已提交
97
  * `B` [线性搜索](src/algorithms/search/linear-search)
H
Henry 已提交
98 99 100
  * `B` [跳转搜索/块搜索](src/algorithms/search/jump-search) - 搜索有序数组
  * `B` [二分查找](src/algorithms/search/binary-search) - 搜索有序数组
  * `B` [插值搜索](src/algorithms/search/interpolation-search) - 搜索均匀分布的有序数组
琚致远 已提交
101
* **排序**
O
Oleksii Trekhleb 已提交
102 103 104 105 106
  * `B` [冒泡排序](src/algorithms/sorting/bubble-sort)
  * `B` [选择排序](src/algorithms/sorting/selection-sort)
  * `B` [插入排序](src/algorithms/sorting/insertion-sort)
  * `B` [堆排序](src/algorithms/sorting/heap-sort)
  * `B` [归并排序](src/algorithms/sorting/merge-sort)
H
Henry 已提交
107
  * `B` [快速排序](src/algorithms/sorting/quick-sort) - in-place (原地) 和 non-in-place 版本
O
Oleksii Trekhleb 已提交
108 109 110
  * `B` [希尔排序](src/algorithms/sorting/shell-sort)
  * `B` [计数排序](src/algorithms/sorting/counting-sort)
  * `B` [基数排序](src/algorithms/sorting/radix-sort)
H
Henry 已提交
111 112 113
* **链表**
  - `B` [正向遍历](src/algorithms/linked-list/traversal)
  - `B` [反向遍历](src/algorithms/linked-list/reverse-traversal)
114
* **树**
O
Oleksii Trekhleb 已提交
115 116
  * `B` [深度优先搜索](src/algorithms/tree/depth-first-search) (DFS)
  * `B` [广度优先搜索](src/algorithms/tree/breadth-first-search) (BFS)
琚致远 已提交
117
* **图**
O
Oleksii Trekhleb 已提交
118 119
  * `B` [深度优先搜索](src/algorithms/graph/depth-first-search) (DFS)
  * `B` [广度优先搜索](src/algorithms/graph/breadth-first-search) (BFS)
H
Henry 已提交
120
  * `B` [克鲁斯克尔演算法](src/algorithms/graph/kruskal) - 寻找加权无向图的最小生成树 (MST)
O
Oleksii Trekhleb 已提交
121 122 123
  * `A` [戴克斯特拉算法](src/algorithms/graph/dijkstra) - 找到图中所有顶点的最短路径
  * `A` [贝尔曼-福特算法](src/algorithms/graph/bellman-ford) - 找到图中所有顶点的最短路径
  * `A` [弗洛伊德算法](src/algorithms/graph/floyd-warshall) - 找到所有顶点对 之间的最短路径
H
Henry 已提交
124
  * `A` [判圈算法](src/algorithms/graph/detect-cycle) - 对于有向图和无向图 (基于 DFS 和不相交集的版本)
125
  * `A` [普林演算法](src/algorithms/graph/prim) - 寻找加权无向图的最小生成树 (MST)
O
Oleksii Trekhleb 已提交
126
  * `A` [拓扑排序](src/algorithms/graph/topological-sorting) - DFS 方法
H
Henry 已提交
127 128 129
  * `A` [关节点](src/algorithms/graph/articulation-points) - Tarjan 算法 (基于 DFS)
  * `A` [](src/algorithms/graph/bridges) - 基于 DFS 的算法
  * `A` [欧拉回径与一笔画问题](src/algorithms/graph/eulerian-path) - Fleury 的算法 - 一次访问每个边
O
Oleksii Trekhleb 已提交
130
  * `A` [哈密顿图](src/algorithms/graph/hamiltonian-cycle) - 恰好访问每个顶点一次
H
Henry 已提交
131
  * `A` [强连通分量](src/algorithms/graph/strongly-connected-components) - Kosaraju 算法
O
Oleksii Trekhleb 已提交
132
  * `A` [旅行推销员问题](src/algorithms/graph/travelling-salesman) - 尽可能以最短的路线访问每个城市并返回原始城市
H
Henry 已提交
133 134
* **加密**
  * `B` [多项式 hash](src/algorithms/cryptography/polynomial-hash) - 基于多项式的 rolling hash 函数
135 136
* **机器学习**
  * `B` [NanoNeuron](https://github.com/trekhleb/nano-neuron) -7个简单的JS函数,说明机器如何实际学习(向前/向后传播)
137
* **未分类**
O
Oleksii Trekhleb 已提交
138 139
  * `B` [汉诺塔](src/algorithms/uncategorized/hanoi-tower)
  * `B` [旋转矩阵](src/algorithms/uncategorized/square-matrix-rotation) - 原地算法
H
Henry 已提交
140 141
  * `B` [跳跃游戏](src/algorithms/uncategorized/jump-game) - 回溯,、动态编程 (自上而下+自下而上) 和贪婪的例子
  * `B` [独特(唯一) 路径](src/algorithms/uncategorized/unique-paths) - 回溯、动态编程和基于 Pascal 三角形的例子
142
  * `B` [雨水收集](src/algorithms/uncategorized/rain-terraces) - 诱捕雨水问题 (动态编程和暴力版本)
H
Henry 已提交
143
  * `B` [递归楼梯](src/algorithms/uncategorized/recursive-staircase) - 计算有共有多少种方法可以到达顶层 (4 种解题方案)
O
Oleksii Trekhleb 已提交
144 145
  * `A` [八皇后问题](src/algorithms/uncategorized/n-queens)
  * `A` [骑士巡逻](src/algorithms/uncategorized/knight-tour)
琚致远 已提交
146 147 148

### 算法范式

H
Henry 已提交
149
算法范式是一种通用方法,基于一类算法的设计。这是比算法更高的抽象,就像算法是比计算机程序更高的抽象。
琚致远 已提交
150

H
Henry 已提交
151
* **BF 算法** - `查找/搜索` 所有可能性并选择最佳解决方案
O
Oleksii Trekhleb 已提交
152 153
  * `B` [线性搜索](src/algorithms/search/linear-search)
  * `B` [雨水收集](src/algorithms/uncategorized/rain-terraces) - 诱导雨水问题
H
Henry 已提交
154
  * `B` [递归楼梯](src/algorithms/uncategorized/recursive-staircase) - 计算有共有多少种方法可以到达顶层 (4 种解题方案)
O
Oleksii Trekhleb 已提交
155 156
  * `A` [最大子数列](src/algorithms/sets/maximum-subarray)
  * `A` [旅行推销员问题](src/algorithms/graph/travelling-salesman) - 尽可能以最短的路线访问每个城市并返回原始城市
H
Henry 已提交
157 158
  * `A` [离散傅里叶变换](src/algorithms/math/fourier-transform) - 把时间信号解析成构成它的频率
* **贪心法** - 在当前选择最佳选项,不考虑以后情况
O
Oleksii Trekhleb 已提交
159 160 161 162 163
  * `B` [跳跃游戏](src/algorithms/uncategorized/jump-game)
  * `A` [背包问题](src/algorithms/sets/knapsack-problem)
  * `A` [戴克斯特拉算法](src/algorithms/graph/dijkstra) - 找到所有图顶点的最短路径
  * `A` [普里姆算法](src/algorithms/graph/prim) - 寻找加权无向图的最小生成树 (MST)
  * `A` [克鲁斯卡尔算法](src/algorithms/graph/kruskal) - 寻找加权无向图的最小生成树 (MST)
H
Henry 已提交
164
* **分治法** - 将问题分成较小的部分,然后解决这些部分
O
Oleksii Trekhleb 已提交
165 166 167 168 169 170 171 172
  * `B` [二分查找](src/algorithms/search/binary-search)
  * `B` [汉诺塔](src/algorithms/uncategorized/hanoi-tower)
  * `B` [杨辉三角形](src/algorithms/math/pascal-triangle)
  * `B` [欧几里得算法](src/algorithms/math/euclidean-algorithm) - 计算最大公约数 (GCD)
  * `B` [归并排序](src/algorithms/sorting/merge-sort)
  * `B` [快速排序](src/algorithms/sorting/quick-sort)
  * `B` [树深度优先搜索](src/algorithms/tree/depth-first-search) (DFS)
  * `B` [图深度优先搜索](src/algorithms/graph/depth-first-search) (DFS)
H
Henry 已提交
173 174
  * `B` [跳跃游戏](src/algorithms/uncategorized/jump-game)
  * `B` [快速算次方](src/algorithms/math/fast-powering)
O
Oleksii Trekhleb 已提交
175 176
  * `A` [排列](src/algorithms/sets/permutations) (有/无重复)
  * `A` [组合](src/algorithms/sets/combinations) (有/无重复)
H
Henry 已提交
177
* **动态编程** - 使用以前找到的子解决方案构建解决方案
O
Oleksii Trekhleb 已提交
178 179 180 181
  * `B` [斐波那契数](src/algorithms/math/fibonacci)
  * `B` [跳跃游戏](src/algorithms/uncategorized/jump-game)
  * `B` [独特路径](src/algorithms/uncategorized/unique-paths)
  * `B` [雨水收集](src/algorithms/uncategorized/rain-terraces) - 疏导雨水问题
H
Henry 已提交
182
  * `B` [递归楼梯](src/algorithms/uncategorized/recursive-staircase) - 计算有共有多少种方法可以到达顶层 (4 种解题方案)
O
Oleksii Trekhleb 已提交
183 184 185 186 187 188 189 190 191
  * `A` [莱温斯坦距离](src/algorithms/string/levenshtein-distance) - 两个序列之间的最小编辑距离
  * `A` [最长公共子序列](src/algorithms/sets/longest-common-subsequence) (LCS)
  * `A` [最长公共子串](src/algorithms/string/longest-common-substring)
  * `A` [最长递增子序列](src/algorithms/sets/longest-increasing-subsequence)
  * `A` [最短公共子序列](src/algorithms/sets/shortest-common-supersequence)
  * `A` [0-1背包问题](src/algorithms/sets/knapsack-problem)
  * `A` [整数拆分](src/algorithms/math/integer-partition)
  * `A` [最大子数列](src/algorithms/sets/maximum-subarray)
  * `A` [贝尔曼-福特算法](src/algorithms/graph/bellman-ford) - 找到所有图顶点的最短路径
H
Henry 已提交
192 193 194
  * `A` [弗洛伊德算法](src/algorithms/graph/floyd-warshall) - 找到所有顶点对之间的最短路径
  * `A` [正则表达式匹配](src/algorithms/string/regular-expression-matching)
* **回溯法** - 类似于 `BF 算法` 试图产生所有可能的解决方案,但每次生成解决方案测试如果它满足所有条件,那么只有继续生成后续解决方案。否则回溯并继续寻找不同路径的解决方案。
O
Oleksii Trekhleb 已提交
195 196
  * `B` [跳跃游戏](src/algorithms/uncategorized/jump-game)
  * `B` [独特路径](src/algorithms/uncategorized/unique-paths)
H
Henry 已提交
197
  * `A` [幂集](src/algorithms/sets/power-set) - 该集合的所有子集
O
Oleksii Trekhleb 已提交
198 199 200
  * `A` [哈密顿图](src/algorithms/graph/hamiltonian-cycle) - 恰好访问每个顶点一次
  * `A` [八皇后问题](src/algorithms/uncategorized/n-queens)
  * `A` [骑士巡逻](src/algorithms/uncategorized/knight-tour)
201
  * `A` [组合求和](src/algorithms/sets/combination-sum) - 从规定的总和中找出所有的组合
H
Henry 已提交
202
* **Branch & Bound** - 记住在回溯搜索的每个阶段找到的成本最低的解决方案,并使用到目前为止找到的成本最小值作为下限。以便丢弃成本大于最小值的解决方案。通常,使用 BFS 遍历以及状态空间树的 DFS 遍历。
琚致远 已提交
203 204 205 206 207 208 209 210

## 如何使用本仓库

**安装依赖**
```
npm install
```

H
Henry 已提交
211 212 213 214 215 216 217 218
**运行 ESLint**

检查代码质量

```
npm run lint
```

琚致远 已提交
219
**执行测试**
H
Henry 已提交
220

琚致远 已提交
221 222 223 224 225 226
```
npm test
```

**按照名称执行测试**
```
227
npm test -- 'LinkedList'
琚致远 已提交
228 229 230 231
```

**Playground**

H
Henry 已提交
232
你可以在 `./src/playground/playground.js` 文件中操作数据结构与算法,并在 `./src/playground/__test__/playground.test.js` 中编写测试。
琚致远 已提交
233

234
然后,只需运行以下命令来测试你的 Playground 是否无误:
琚致远 已提交
235 236

```
237
npm test -- 'playground'
琚致远 已提交
238 239 240 241 242 243 244 245 246 247 248 249
```

## 有用的信息

### 引用

[▶ YouTube](https://www.youtube.com/playlist?list=PLLXdhg_r2hKA7DPDsunoDZ-Z769jWn4R8)

### 大O符号

大O符号中指定的算法的增长顺序。

T
Tien Do 已提交
250
![Big O graphs](./assets/big-o-graph.png)
琚致远 已提交
251 252 253 254 255

源: [Big O Cheat Sheet](http://bigocheatsheet.com/).

以下是一些最常用的 大O标记法 列表以及它们与不同大小输入数据的性能比较。

O
Oleksii Trekhleb 已提交
256
| 大O标记法      | 计算10个元素                 | 计算100个元素                 | 计算1000个元素                  |
琚致远 已提交
257 258 259 260
| -------------- | ---------------------------- | ----------------------------- | ------------------------------- |
| **O(1)**       | 1                            | 1                             | 1                               |
| **O(log N)**   | 3                            | 6                             | 9                               |
| **O(N)**       | 10                           | 100                           | 1000                            |
O
Oleksii Trekhleb 已提交
261
| **O(N log N)** | 30                           | 600                           | 9000                            |
琚致远 已提交
262 263 264 265 266 267
| **O(N^2)**     | 100                          | 10000                         | 1000000                         |
| **O(2^N)**     | 1024                         | 1.26e+29                      | 1.07e+301                       |
| **O(N!)**      | 3628800                      | 9.3e+157                      | 4.02e+2567                      |

### 数据结构操作的复杂性

H
Henry 已提交
268 269 270 271 272 273 274
| 数据结构       |  连接  |  查找  |  插入  |  删除  | 备注 |
| -------------- | :----: | :----: | :----: | :----: | ---- |
| **数组**       |   1    |   n    |   n    |   n    |      |
| **栈**         |   n    |   n    |   1    |   1    |      |
| **队列**       |   n    |   n    |   1    |   1    |      |
| **链表**       |   n    |   n    |   1    |   1    |      |
| **哈希表**     |   -    |   n    |   n    |   n    | 在完全哈希函数情况下,复杂度是 O(1) |
O
Oleksii Trekhleb 已提交
275 276
| **二分查找树** |   n    |   n    |   n    |   n    | 在平衡树情况下,复杂度是 O(log(n)) |
| **B 树**       | log(n) | log(n) | log(n) | log(n) |      |
H
Henry 已提交
277
| **红黑树**     | log(n) | log(n) | log(n) | log(n) |      |
O
Oleksii Trekhleb 已提交
278
| **AVL 树**     | log(n) | log(n) | log(n) | log(n) |      |
H
Henry 已提交
279
| **布隆过滤器** |   -    |   1    |   1    | - | 存在一定概率的判断错误(误判成存在) |
琚致远 已提交
280 281 282

### 数组排序算法的复杂性

O
Oleksii Trekhleb 已提交
283
| 名称                  | 最优      | 平均      | 最坏          | 内存      | 稳定      | 备注                  |
H
Henry 已提交
284
| --------------------- | :-------: | :-------: | :-----------: | :-------: | :-------: | --------------------- |
O
Oleksii Trekhleb 已提交
285 286 287 288 289
| **冒泡排序**          | n         | n^2       | n^2           | 1         | Yes       |                       |
| **插入排序**          | n         | n^2       | n^2           | 1         | Yes       |                       |
| **选择排序**          | n^2       | n^2       | n^2           | 1         | No        |                       |
| **堆排序**            | n log(n)  | n log(n)  | n log(n)      | 1         | No        |                       |
| **归并排序**          | n log(n)  | n log(n)  | n log(n)      | n         | Yes       |                       |
H
Henry 已提交
290
| **快速排序**          | n log(n)  | n log(n)  | n^2           | log(n)    | No        | 在 in-place 版本下,内存复杂度通常是 O(log(n)) |
O
Oleksii Trekhleb 已提交
291 292 293
| **希尔排序**          | n log(n)  | 取决于差距序列   | n (log(n))^2  | 1         | No        |  |
| **计数排序**          | n + r     | n + r     | n + r         | n + r     | Yes       | r - 数组里最大的数    |
| **基数排序**          | n * k     | n * k     | n * k         | n + k     | Yes       | k - 最长 key 的升序   |