目录
定义
快速排序是一个知名度极高的排序算法,其对于大数据的优秀排序性能和相同复杂度算法中相对简单的实现使它注定得到比其他算法更多的宠爱。
算法描述
- 从数列中挑出一个元素,称为"基准"(pivot),
- 重新排序数列,所有比基准值小的元素摆放在基准前面,所有比基准值大的元素摆在基准后面(相同的数可以到任何一边)。在这个分区结束之后,该基准就处于数列的中间位置。这个称为分区(partition)操作。
- 递归地(recursively)把小于基准值元素的子数列和大于基准值元素的子数列排序。
动图演示
稳定性
快速排序并不是稳定的。这是因为我们无法保证相等的数据按顺序被扫描到和按顺序存放。
适用场景
快速排序在大多数情况下都是适用的,尤其在数据量大的时候性能优越性更加明显。但是在必要的时候,需要考虑下优化以提高其在最坏情况下的性能。
代码实现
Kotlin实现
/**
* 快速排序
*/
private fun quickSort(arr: IntArray) {
quickSort(arr, 0, arr.size - 1)
}
private fun quickSort(arr: IntArray, low: Int, high: Int) {
if (low >= high) return
// 将数组分为两部分
var pivot = partition(arr, low, high)
// 递归排序左子数组
quickSort(arr, low, pivot - 1)
// 递归排序右子数组
quickSort(arr, pivot + 1, high)
}
private fun partition(arr: IntArray, low: Int, high: Int): Int {
var low = low
var high = high
// 基准
var pivot = arr[low]
while (low < high) {
while (low < high && arr[high] >= pivot) {
--high
}
// 交换比基准大的记录到左端
arr[low] = arr[high]
while (low < high && arr[low] <= pivot) {
++low
}
// 交换比基准小的记录到右端
arr[high] = arr[low]
}
// 扫描完成,基准到位
arr[low] = pivot
// 返回的是基准的位置
return low
}
Java实现
/**
* 快速排序
*/
public static void quickSort(int[] arr){
qsort(arr, 0, arr.length-1);
}
private static void qsort(int[] arr, int low, int high){
if (low >= high)
return;
int pivot = partition(arr, low, high); //将数组分为两部分
qsort(arr, low, pivot-1); //递归排序左子数组
qsort(arr, pivot+1, high); //递归排序右子数组
}
private static int partition(int[] arr, int low, int high){
int pivot = arr[low]; //基准
while (low < high){
while (low < high && arr[high] >= pivot) --high;
arr[low]=arr[high]; //交换比基准大的记录到左端
while (low < high && arr[low] <= pivot) ++low;
arr[high] = arr[low]; //交换比基准小的记录到右端
}
//扫描完成,基准到位
arr[low] = pivot;
//返回的是基准的位置
return low;
}