hello-algo/docs/chapter_sorting/quick_sort.md

369 lines
11 KiB
Markdown
Raw Normal View History

# 快速排序
2022-11-22 17:47:26 +08:00
「快速排序 Quick Sort」是一种基于“分治思想”的排序算法速度很快、应用很广。
2022-11-23 03:56:25 +08:00
2023-01-09 22:39:30 +08:00
快速排序的核心操作为「哨兵划分」,其目标为:选取数组某个元素为 **基准数**,将所有小于基准数的元素移动至其左边,大于基准数的元素移动至其右边。「哨兵划分」的实现流程为:
2022-11-23 03:56:25 +08:00
1. 以数组最左端元素作为基准数,初始化两个指针 `i` , `j` 指向数组两端;
2. 设置一个循环,每轮中使用 `i` / `j` 分别寻找首个比基准数大 / 小的元素,并交换此两元素;
3. 不断循环步骤 `2.` ,直至 `i` , `j` 相遇时跳出,最终把基准数交换至两个子数组的分界线;
2023-01-09 22:39:30 +08:00
「哨兵划分」执行完毕后,原数组被划分成两个部分,即 **左子数组****右子数组**,且满足 **左子数组任意元素 < 基准数 < 右子数组任意元素**。因此,接下来我们只需要排序两个子数组即可。
2022-11-23 03:56:25 +08:00
2023-02-22 00:57:43 +08:00
=== "<1>"
2023-02-26 19:22:46 +08:00
![哨兵划分步骤](quick_sort.assets/pivot_division_step1.png)
2023-01-15 23:32:58 +08:00
2023-02-22 00:57:43 +08:00
=== "<2>"
2022-11-23 03:56:25 +08:00
![pivot_division_step2](quick_sort.assets/pivot_division_step2.png)
2023-01-15 23:32:58 +08:00
2023-02-22 00:57:43 +08:00
=== "<3>"
2022-11-23 03:56:25 +08:00
![pivot_division_step3](quick_sort.assets/pivot_division_step3.png)
2023-01-15 23:32:58 +08:00
2023-02-22 00:57:43 +08:00
=== "<4>"
2022-11-23 03:56:25 +08:00
![pivot_division_step4](quick_sort.assets/pivot_division_step4.png)
2023-01-15 23:32:58 +08:00
2023-02-22 00:57:43 +08:00
=== "<5>"
2022-11-23 03:56:25 +08:00
![pivot_division_step5](quick_sort.assets/pivot_division_step5.png)
2023-01-15 23:32:58 +08:00
2023-02-22 00:57:43 +08:00
=== "<6>"
2022-11-23 03:56:25 +08:00
![pivot_division_step6](quick_sort.assets/pivot_division_step6.png)
2023-01-15 23:32:58 +08:00
2023-02-22 00:57:43 +08:00
=== "<7>"
2022-11-23 03:56:25 +08:00
![pivot_division_step7](quick_sort.assets/pivot_division_step7.png)
2023-01-15 23:32:58 +08:00
2023-02-22 00:57:43 +08:00
=== "<8>"
2022-11-23 03:56:25 +08:00
![pivot_division_step8](quick_sort.assets/pivot_division_step8.png)
2023-01-15 23:32:58 +08:00
2023-02-22 00:57:43 +08:00
=== "<9>"
2022-11-23 03:56:25 +08:00
![pivot_division_step9](quick_sort.assets/pivot_division_step9.png)
!!! note "快速排序的分治思想"
哨兵划分的实质是将 **一个长数组的排序问题** 简化为 **两个短数组的排序问题**
2022-11-23 03:56:25 +08:00
=== "Java"
```java title="quick_sort.java"
[class]{QuickSort}-[func]{swap}
[class]{QuickSort}-[func]{partition}
2022-11-23 03:56:25 +08:00
```
=== "C++"
```cpp title="quick_sort.cpp"
2023-02-08 20:30:05 +08:00
[class]{QuickSort}-[func]{swap}
2023-02-08 20:30:05 +08:00
[class]{QuickSort}-[func]{partition}
```
=== "Python"
```python title="quick_sort.py"
[class]{QuickSort}-[func]{partition}
```
2022-12-03 01:31:29 +08:00
=== "Go"
```go title="quick_sort.go"
[class]{quickSort}-[func]{partition}
2022-12-03 01:31:29 +08:00
```
=== "JavaScript"
2023-02-08 19:45:06 +08:00
```javascript title="quick_sort.js"
2023-02-08 20:30:05 +08:00
[class]{QuickSort}-[func]{swap}
[class]{QuickSort}-[func]{partition}
2022-12-03 01:31:29 +08:00
```
=== "TypeScript"
```typescript title="quick_sort.ts"
2023-02-08 20:30:05 +08:00
[class]{QuickSort}-[func]{swap}
2022-12-03 01:31:29 +08:00
2023-02-08 20:30:05 +08:00
[class]{QuickSort}-[func]{partition}
2022-12-03 01:31:29 +08:00
```
=== "C"
```c title="quick_sort.c"
[class]{quickSort}-[func]{partition}
2022-12-03 01:31:29 +08:00
```
=== "C#"
```csharp title="quick_sort.cs"
[class]{QuickSort}-[func]{swap}
2022-12-23 15:42:02 +08:00
[class]{QuickSort}-[func]{partition}
2022-12-03 01:31:29 +08:00
```
2023-01-08 19:41:05 +08:00
=== "Swift"
```swift title="quick_sort.swift"
2023-02-08 20:30:05 +08:00
[class]{}-[func]{swap}
2023-01-08 19:41:05 +08:00
2023-02-08 20:30:05 +08:00
[class]{}-[func]{partition}
2023-01-08 19:41:05 +08:00
```
2023-02-01 22:03:04 +08:00
=== "Zig"
```zig title="quick_sort.zig"
[class]{QuickSort}-[func]{swap}
2023-02-01 22:03:04 +08:00
[class]{QuickSort}-[func]{partition}
2023-02-01 22:03:04 +08:00
```
## 算法流程
2022-11-23 03:56:25 +08:00
2023-01-09 22:39:30 +08:00
1. 首先,对数组执行一次「哨兵划分」,得到待排序的 **左子数组****右子数组**
2022-11-23 03:56:25 +08:00
2. 接下来,对 **左子数组****右子数组** 分别 **递归执行**「哨兵划分」……
2023-01-09 22:39:30 +08:00
3. 直至子数组长度为 1 时 **终止递归**,即可完成对整个数组的排序;
2022-11-23 03:56:25 +08:00
观察发现,快速排序和「二分查找」的原理类似,都是以对数阶的时间复杂度来缩小处理区间。
![快速排序流程](quick_sort.assets/quick_sort_overview.png)
2022-11-23 15:50:59 +08:00
2022-11-23 03:56:25 +08:00
=== "Java"
2022-11-23 21:39:39 +08:00
```java title="quick_sort.java"
[class]{QuickSort}-[func]{quickSort}
2022-11-23 03:56:25 +08:00
```
=== "C++"
```cpp title="quick_sort.cpp"
2023-02-08 04:17:26 +08:00
[class]{QuickSort}-[func]{quickSort}
```
=== "Python"
```python title="quick_sort.py"
[class]{QuickSort}-[func]{quick_sort}
```
2022-12-03 01:31:29 +08:00
=== "Go"
```go title="quick_sort.go"
[class]{quickSort}-[func]{quickSort}
2022-12-03 01:31:29 +08:00
```
=== "JavaScript"
2023-02-08 04:27:55 +08:00
```javascript title="quick_sort.js"
2023-02-08 19:45:06 +08:00
[class]{QuickSort}-[func]{quickSort}
2022-12-03 01:31:29 +08:00
```
=== "TypeScript"
```typescript title="quick_sort.ts"
2023-02-08 19:45:06 +08:00
[class]{QuickSort}-[func]{quickSort}
2022-12-03 01:31:29 +08:00
```
=== "C"
```c title="quick_sort.c"
[class]{quickSort}-[func]{quickSort}
2022-12-03 01:31:29 +08:00
```
=== "C#"
```csharp title="quick_sort.cs"
[class]{QuickSort}-[func]{quickSort}
2022-12-03 01:31:29 +08:00
```
2023-01-08 19:41:05 +08:00
=== "Swift"
```swift title="quick_sort.swift"
2023-02-08 20:30:05 +08:00
[class]{}-[func]{quickSort}
2023-01-08 19:41:05 +08:00
```
2023-02-01 22:03:04 +08:00
=== "Zig"
```zig title="quick_sort.zig"
[class]{QuickSort}-[func]{quickSort}
2023-02-01 22:03:04 +08:00
```
## 算法特性
2022-11-23 03:56:25 +08:00
2023-03-26 04:47:05 +08:00
**时间复杂度 $O(n \log n)$** :平均情况下,哨兵划分的递归层数为 $\log n$ ,每层中的总循环数为 $n$ ,总体使用 $O(n \log n)$ 时间。
最差情况下,每轮哨兵划分操作都将长度为 $n$ 的数组划分为长度为 $0$ 和 $n - 1$ 的两个子数组,此时递归层数达到 $n$ 层,每层中的循环数为 $n$ ,总体使用 $O(n^2)$ 时间,因此是“非稳定排序”。
2022-11-23 03:56:25 +08:00
**空间复杂度 $O(n)$** :输入数组完全倒序下,达到最差递归深度 $n$ 。由于未借助辅助数组空间,因此是“原地排序”。
2022-11-23 03:56:25 +08:00
**非稳定排序**:哨兵划分最后一步可能会将基准数交换至相等元素的右边。
2022-11-23 03:56:25 +08:00
## 快排为什么快?
2022-11-23 03:56:25 +08:00
2023-01-09 22:39:30 +08:00
从命名能够看出,快速排序在效率方面一定“有两把刷子”。快速排序的平均时间复杂度虽然与「归并排序」和「堆排序」一致,但实际 **效率更高**,这是因为:
2022-11-23 03:56:25 +08:00
2023-01-09 22:39:30 +08:00
- **出现最差情况的概率很低**:虽然快速排序的最差时间复杂度为 $O(n^2)$ ,不如归并排序,但绝大部分情况下,快速排序可以达到 $O(n \log n)$ 的复杂度。
- **缓存使用效率高**:哨兵划分操作时,将整个子数组加载入缓存中,访问元素效率很高。而诸如「堆排序」需要跳跃式访问元素,因此不具有此特性。
- **复杂度的常数系数低**:在提及的三种算法中,快速排序的 **比较**、**赋值**、**交换** 三种操作的总体数量最少(类似于「插入排序」快于「冒泡排序」的原因)。
2022-11-23 03:56:25 +08:00
## 基准数优化
2022-11-23 03:56:25 +08:00
2023-01-09 22:39:30 +08:00
**普通快速排序在某些输入下的时间效率变差**。举个极端例子,假设输入数组是完全倒序的,由于我们选取最左端元素为基准数,那么在哨兵划分完成后,基准数被交换至数组最右端,从而 **左子数组长度为 $n - 1$、右子数组长度为 $0$** 。这样进一步递归下去,**每轮哨兵划分后的右子数组长度都为 $0$** ,分治策略失效,快速排序退化为「冒泡排序」了。
2022-11-23 03:56:25 +08:00
2023-01-09 22:39:30 +08:00
为了尽量避免这种情况发生,我们可以优化一下基准数的选取策略。首先,在哨兵划分中,我们可以 **随机选取一个元素作为基准数**。但如果运气很差,每次都选择到比较差的基准数,那么效率依然不好。
2022-11-23 03:56:25 +08:00
进一步地,我们可以在数组中选取 3 个候选元素(一般为数组的首、尾、中点元素),**并将三个候选元素的中位数作为基准数**,这样基准数“既不大也不小”的概率就大大提升了。当然,如果数组很长的话,我们也可以选取更多候选元素,来进一步提升算法的稳健性。采取该方法后,时间复杂度劣化至 $O(n^2)$ 的概率极低。
2022-11-23 03:56:25 +08:00
=== "Java"
2022-11-23 21:39:39 +08:00
```java title="quick_sort.java"
[class]{QuickSortMedian}-[func]{medianThree}
[class]{QuickSortMedian}-[func]{partition}
2022-11-23 03:56:25 +08:00
```
=== "C++"
```cpp title="quick_sort.cpp"
2023-02-08 04:17:26 +08:00
[class]{QuickSortMedian}-[func]{medianThree}
2023-02-08 04:17:26 +08:00
[class]{QuickSortMedian}-[func]{partition}
```
=== "Python"
```python title="quick_sort.py"
[class]{QuickSortMedian}-[func]{median_three}
[class]{QuickSortMedian}-[func]{partition}
```
2022-12-03 01:31:29 +08:00
=== "Go"
```go title="quick_sort.go"
[class]{quickSortMedian}-[func]{medianThree}
[class]{quickSortMedian}-[func]{partition}
2022-12-03 01:31:29 +08:00
```
=== "JavaScript"
2023-02-08 04:27:55 +08:00
```javascript title="quick_sort.js"
2023-02-08 19:45:06 +08:00
[class]{QuickSortMedian}-[func]{medianThree}
[class]{QuickSortMedian}-[func]{partition}
2022-12-03 01:31:29 +08:00
```
=== "TypeScript"
```typescript title="quick_sort.ts"
2023-02-08 19:45:06 +08:00
[class]{QuickSortMedian}-[func]{medianThree}
2022-12-03 01:31:29 +08:00
2023-02-08 19:45:06 +08:00
[class]{QuickSortMedian}-[func]{partition}
2022-12-03 01:31:29 +08:00
```
=== "C"
```c title="quick_sort.c"
[class]{quickSortMedian}-[func]{medianThree}
2022-12-03 01:31:29 +08:00
[class]{quickSortMedian}-[func]{partition}
2022-12-03 01:31:29 +08:00
```
=== "C#"
```csharp title="quick_sort.cs"
[class]{QuickSortMedian}-[func]{medianThree}
2022-12-03 01:31:29 +08:00
[class]{QuickSortMedian}-[func]{partition}
2022-12-03 01:31:29 +08:00
```
2023-01-08 19:41:05 +08:00
=== "Swift"
```swift title="quick_sort.swift"
2023-02-08 20:30:05 +08:00
[class]{}-[func]{medianThree}
2023-01-08 19:41:05 +08:00
2023-02-08 20:30:05 +08:00
[class]{}-[func]{partitionMedian}
2023-01-08 19:41:05 +08:00
```
2023-02-01 22:03:04 +08:00
=== "Zig"
```zig title="quick_sort.zig"
[class]{QuickSortMedian}-[func]{medianThree}
2023-02-01 22:03:04 +08:00
[class]{QuickSortMedian}-[func]{partition}
2023-02-01 22:03:04 +08:00
```
## 尾递归优化
2022-11-23 03:56:25 +08:00
**普通快速排序在某些输入下的空间效率变差**。仍然以完全倒序的输入数组为例,由于每轮哨兵划分后右子数组长度为 0 ,那么将形成一个高度为 $n - 1$ 的递归树,此时使用的栈帧空间大小劣化至 $O(n)$ 。
2022-11-23 03:56:25 +08:00
为了避免栈帧空间的累积,我们可以在每轮哨兵排序完成后,判断两个子数组的长度大小,仅递归排序较短的子数组。由于较短的子数组长度不会超过 $\frac{n}{2}$ ,因此这样做能保证递归深度不超过 $\log n$ ,即最差空间复杂度被优化至 $O(\log n)$ 。
=== "Java"
2022-11-23 21:39:39 +08:00
```java title="quick_sort.java"
[class]{QuickSortTailCall}-[func]{quickSort}
2022-12-01 18:28:57 +08:00
```
=== "C++"
```cpp title="quick_sort.cpp"
2023-02-08 04:17:26 +08:00
[class]{QuickSortTailCall}-[func]{quickSort}
2022-11-23 03:56:25 +08:00
```
=== "Python"
```python title="quick_sort.py"
[class]{QuickSortTailCall}-[func]{quick_sort}
```
2022-12-03 01:31:29 +08:00
=== "Go"
```go title="quick_sort.go"
[class]{quickSortTailCall}-[func]{quickSort}
2022-12-03 01:31:29 +08:00
```
=== "JavaScript"
2023-02-08 04:27:55 +08:00
```javascript title="quick_sort.js"
2023-02-08 19:45:06 +08:00
[class]{QuickSortTailCall}-[func]{quickSort}
2022-12-03 01:31:29 +08:00
```
=== "TypeScript"
```typescript title="quick_sort.ts"
2023-02-08 19:45:06 +08:00
[class]{QuickSortTailCall}-[func]{quickSort}
2022-12-03 01:31:29 +08:00
```
=== "C"
```c title="quick_sort.c"
[class]{quickSortTailCall}-[func]{quickSort}
2022-12-03 01:31:29 +08:00
```
=== "C#"
```csharp title="quick_sort.cs"
[class]{QuickSortTailCall}-[func]{quickSort}
2022-12-03 01:31:29 +08:00
```
2023-01-08 19:41:05 +08:00
=== "Swift"
```swift title="quick_sort.swift"
2023-02-08 20:30:05 +08:00
[class]{}-[func]{quickSortTailCall}
2023-01-08 19:41:05 +08:00
```
2023-02-01 22:03:04 +08:00
=== "Zig"
```zig title="quick_sort.zig"
[class]{QuickSortTailCall}-[func]{quickSort}
2023-02-01 22:03:04 +08:00
```
2023-03-26 22:02:37 +08:00
!!! question "哨兵划分中“从右往左查找”与“从左往右查找”的顺序可以交换吗?"
不行,当我们以最左端元素为基准数时,必须先“从右往左查找”再“从左往右查找”。这个结论有些反直觉,我们来剖析一下原因。
哨兵划分 `partition()` 的最后一步是交换 `nums[left]``nums[i]` ,完成交换后,基准数左边的元素都 `<=` 基准数,**这就要求最后一步交换前 `nums[left] >= nums[i]` 必须成立**。假设我们先“从左往右查找”,那么如果找不到比基准数更小的元素,**则会在 `i == j` 时跳出循环,此时可能 `nums[j] == nums[i] > nums[left]`** ;也就是说,此时最后一步交换操作会把一个比基准数更大的元素交换至数组最左端,导致哨兵划分失败。
举个例子,给定数组 `[0, 0, 0, 0, 1]` ,如果先“从左向右查找”,哨兵划分后数组为 `[1, 0, 0, 0, 0]` ,这个结果是不对的。
再深想一步,如果我们选择 `nums[right]` 为基准数,那么正好反过来,必须先“从左往右查找”。