背景知识
快速排序和归并排序都基于递归的思想。其中归并排序每次将输入平均划分为两部分,因此时间复杂度为O(nlogn)。而快速排序根据选择的 pivot将其余的数据划分为大于s[pivot]和小于s[pivot]的,只有当s[pivot]恰好是中位数时,时间复杂度才是O(nlogn),最坏情况下, 每次选择的pivot都位于边缘,如下图所示。这样时间复杂度就退化为$O(n^2)$
以上的分析是基于内存模型和大 O表示法的,而在实际运行过程中,由于归并排序归并需要的时间比快速排序时间长,因此时间的比较可能出现不一样的结果。本文将对其实际性能进行对比。本实验基于java 1.8。
实验结果
数组长度 | 初始化情况 | 快速排序 | 归并排序 | System |
---|---|---|---|---|
1e6 | 有序 | StackOverflow | 496 | 5 |
1e6 | 无序 | 153 | 663 | 180 |
1e8 | 有序 | StackOverflow | 496 | 131 |
1e6 | 无序 | 17380 | too long | 17471 |
时间单位:ms
结果分析
可以看出,当数组是有序的时候,我们实现的快速排序算法性能会急剧下降,具体原因文章开头已经介绍了。对于这种情况,有以下几种解决方式:
排序之前对数组进行随机排列,时间复杂度为O(n)。
检查数组之前是否是有序的,在java Arrays.sort()中,如果数组长度超过286,且有序的子片段不超过67,则通过归并排序进行合并。
1 | public static <T> void sort(T[] a, Comparator<? super T> c) {} |
1 | public static void sort(int[] a) { |
核心代码
快速排序
1 | public class QuickSort { |
归并排序
1 | public class MergeSort { |