1. 原理
快速排序,说白了就是给基准数据找其正确索引位置的过程.
如下图所示,假设最开始的基准数据为数组第一个元素23,则首先用一个临时变量去存储基准数据,即tmp=23;然后分别从数组的两端扫描数组,设两个指示标志:low指向起始位置,high指向末尾.
首先从后半部分开始,如果扫描到的值大于基准数据就让high减1,如果发现有元素比该基准数据的值小(如上图中18<=tmp),就将high位置的值赋值给low位置 ,结果如下:
然后开始从前往后扫描,如果扫描到的值小于基准数据就让low加1,如果发现有元素大于基准数据的值(如上图46=>tmp),就再将low位置的值赋值给high位置的值,指针移动并且数据交换后的结果如下:
然后再开始从后向前扫描,原理同上,发现上图11<=tmp,则将low位置的值赋值给high位置的值,结果如下:
然后再开始从前往后遍历,直到low=high结束循环,此时low或high的下标就是基准数据23在该数组中的正确索引位置.如下图所示.
这样一遍走下来,可以很清楚的知道,其实快速排序的本质就是把基准数大的都放在基准数的右边,把比基准数小的放在基准数的左边,这样就找到了该数据在数组中的正确位置.
以后采用递归的方式分别对前半部分和后半部分排序,当前半部分和后半部分均有序时该数组就自然有序了。
一些小结论
从上面的过程中可以看到:
①先从队尾开始向前扫描且当low < high时,如果a[high] > tmp,则high–,但如果a[high] < tmp,则将high的值赋值给low,即arr[low] = a[high],同时要转换数组扫描的方式,即需要从队首开始向队尾进行扫描了
②同理,当从队首开始向队尾进行扫描时,如果a[low] < tmp,则low++,但如果a[low] > tmp了,则就需要将low位置的值赋值给high位置,即arr[low] = arr[high],同时将数组扫描方式换为由队尾向队首进行扫描.
③不断重复①和②,知道low>=high时(其实是low=high),low或high的位置就是该基准数据在数组中的正确索引位置.
按照上诉理论我写的代码如下:
2. 算法分析
排序方式:in-place
时间复杂度:
- 最佳情况:T(n) = O(nlog n)
- 最差情况:T(n) = O(n2)
- 平均情况:T(n) = O(nlog n)
空间复杂度:
- 平均:O(log n)
- 最差: O(n)
3. 代码
public static void main(String[] args) {
long[] arrs = RandomArr.createLongArr(20, 0, 200);
long[] arrs2 = Arrays.copyOf(arrs, arrs.length);
Arrays.stream(arrs).forEach(l -> {
System.out.print(l + " ");
});
System.out.println();
QuickSort(arrs, 0, arrs.length - 1);
Arrays.stream(arrs).forEach(l -> {
System.out.print(l + " ");
});
System.out.println();
Arrays.stream(arrs2).sorted().forEach(l -> {
System.out.print(l + " ");
});
}
public static void QuickSort(long[] arrs, int begin, int end) {
if (begin < end) {
int index = getIndex(arrs, begin, end);
System.out.println(index);
QuickSort(arrs, begin, index - 1);
QuickSort(arrs, index + 1, end);
}
}
public static int getIndex(long[] arrs, int begin, int end) {
// 已开始位置的元素为基准
long tmp = arrs[begin];
while (begin < end) {
// 如果后面元素大于基准元素,就前移索引
while (begin < end && arrs[end] >= tmp) {
end--;
}
// 如果后面元素小于基准元素,则把该元素放到基准元素位置
arrs[begin] = arrs[end];
// 从前往后扫描,如果当前元素小于基准元素则前移索引
while (begin < end && arrs[begin] <= tmp) {
begin++;
}
// 如果当前元素大于基准元素则把当前元素放到后面
arrs[end] = arrs[begin];
}
arrs[begin] = tmp;
return begin;
}