我们先看看维基百科的解释:
快速排序(英语:QuickSort),又称划分交换排序(partition-exchange sort),简称快排,一种排序算法,最早由东尼·霍尔提出。在平均状况下,排序 n 个项目要 O(nlogn) 次比较。在最坏状况下则需要 O(n^2) 次比较,但这种状况并不常见。事实上,快速排序通常明显比其他算法更快,因为它的内部循环(inner loop)可以在大部分的架构上很有效率地达成。
平均情况下快速排序的时间复杂度是Θ(𝑛log𝑛),最坏情况是n²,但通过随机算法可以避免最坏情况。由于递归调用,快排的空间复杂度是Θ(log𝑛)。
快速排序的算法思想是分而治之,将一个大的待排序列,分成两个子序列,然后采用递归的方式,依次将子序列也分成更小的子序列,依次进行,最后得到排序好的序列。算法的实现主要分成三步
- 找到基准点:
- 排列序列,将比基准点小的放在左边的子序列,将比基准点大的放在右边的子序列;
- 采用递归,依次重新选取基准点,在重复进行 1,2 步骤,得到最终的顺序序列
public class QuickSort {
public static void main(String[] args) {
int[] array = new int[]{2, 3, 1, 4, 7, 8, 3, 5, 2, 6, 8, 9, 1};
quickSort(array, 0, array.length - 1);
for (int i = 0; i < array.length; i++) {
System.out.print(array[i] + " ");
}
}
/**
* 递归排序
*
* @param array 待排序列
* @param left 左边起始位置
* @param right 右边结束位置
*/
private static void quickSort(int[] array, int left, int right) {
if (left < right) {
//根据基准点,找到分隔左右子序列的位置索引
int position = position(array, left, right);
//分别进行左右的递归
quickSort(array, left, position - 1);
quickSort(array, position + 1, right);
}
}
/**
* 找到中间
*
* @param array 待排序列
* @param left 左边起始位置
* @param right 右边结束位置
* @return
*/
private static int position(int[] array, int left, int right) {
//找到基准点, 这里使用的是序列的第一个元素
int base = array[left];
while (left < right) {
while (right > left && array[right] >= base) {
right--;
}
//交互位置
array[left] = array[right];
while (left < right && array[left] <= base) {
left++;
}
//交互位置
array[right] = array[left];
}
//此时 left 与 right 是相等的
array[left] = base;
return left;
}
}
运行结果:
1 1 2 2 3 3 4 5 6 7 8 8 9
从运行的结果我们看到,已经正常的排序结束了,说明这个算法已经满足了我们的要求,而且详细的代码分析也已经加上了注释,我想大家应该都能看懂。只要记住核心的几个点就可以了,这里我在重复说明一下:
- 先找基准点 base;
- 比较大小,比 base 小的放在左边序列,比 base 大的放在右边序列;
- 递归左右序列。
注意上面内部的两个 while
循环,这里是使用类似两个指针,分别从序列的左右两个端点开始往中间进行遍历,主要进行的第二步比较和赋值的操作。
排序过程中,对尚未确定最终位置的所有元素进行一遍处理称为一“趟”。下列序列中,不可能是快速排序第二趟结果的是:(D)
A.5, 2, 16, 12, 28, 60, 32, 72 B.2, 16, 5, 28, 12, 60, 32, 72 C.2, 12, 16, 5, 28, 32, 72, 60 D.5, 2, 12, 28, 16, 32, 72, 60
先找第一个中间元素,满足左边的比他小,右边比他大,第二趟就是看第一个元素分割的两半,再同样找是否在子序列中,有一个中间元素满足左边的都比他小,右边的比他大;