快速排序是一种比较常用的排序算法,它的原理是先在待排序的区间中,找一个基准值,再遍历整个待排序区间,将比基准值小(可以等于)的值放到基准值的左边,将比基准值大(也可以包含相等)的值放在基准值的右边,再对分出的两组(两边的数据)按照同样的方法进行操作,最终将会得到有序的数。
我们可以利用递归、非递归的方法,写出快速排序的算法代码。
递归实现
用递归方法实现就是在其中主要的就是分组的过程,再完成递归即可实现排序。
下面介绍几种分组方法。
//交换
public void swap(int[] arr,int i,int j) {
int tmp = arr[i];
arr[i] = arr[j];
arr[j] = tmp;
}
public int partition1(int[] arr, int left, int right){
//基准值
int key = arr[left];
int start = left;
while(left < right){
//一定要先从后向前找第一个小于key的数据位置, 否则数据的位置会产生错误
while(left < right && arr[right] >= key) {
right--;
}
//从前向后找第一个大于key的数据位置
while(left < right &&arr[left] <= key) {
left++;
}
//交换
swap(arr, left, right);
}
//把基准值和相遇的位置的数据进行交换
//错误: arr[left] = key; 相遇位置的数据被覆盖
swap(arr, left, start);
return left;
}
public int partition2(int[] arr, int left, int right){
//获取基准值
int key = arr[left];
//挖坑填坑
while(left < right) {
//从右边找第一个小于key的数据,填左边的坑
while(left < right && arr[right] >= key) {
right--;
}
//填坑
arr[left] = arr[right];
//从左边找第一个大于key的数据,填右边的坑
while(left < right && arr[left] <= key) {
left++;
}
//填坑
arr[right] = arr[left];
}
//填基准值到中间相遇的位置
arr[left] = key;
return left;
}
//三数取中法: 为了让数据分组更加均衡
public int getMid(int[] arr, int left, int right){
// 从 arr[left], arr[mid], arr[right],中选一个中间值
int mid = (left + right) / 2;
if(arr[mid] > arr[left]){
if(arr[mid] < arr[right]) {
return mid;
} else {
if(arr[left] > arr[right]) {
return left;
} else {
return right;
}
}
} else {
if(arr[mid] > arr[right]) {
return mid;
} else {
// mid <= left, <= right
if(arr[left] < arr[right]) {
return left;
} else {
return right;
}
}
}
}
public int partition3(int[] arr, int left, int right){
//三数取中
int mid = getMid(arr, left, right);
swap(arr, left, mid);
int key = arr[left];
int prev = left; //最后一个小于key的位置
int cur = left + 1; //下一个小于key的位置
while(cur <= right){
//如果cur找到下一个小于key的位置,并且prev 和 cur之间有大于key的值,则交换prev, cur的值
if(arr[cur] < key && ++prev != cur) {
swap(arr, prev, cur);
}
cur++;
}
swap(arr, left, prev);
return prev;
}
以上是三种分组的方法,都是符合快速排序的原理的,下面来看看,用递归的方法,是如何实现快速排序的算法的,代码如下:
/*
快速排序
时间复杂度:O(nlogn)
空间复杂度:O(logn)
稳定性:不稳定
*/
public void quickSort(int[] arr,int left,int right) {
if(left < right){
//int mid = partition1(arr, left, right);
//int mid = partition2(arr, left, right);
int mid = partition3(arr, left, right);
quickSort(arr, left, mid-1);
quickSort(arr, mid+1, right);
}
}
递归的实现,我认为是比较好理解的。
非递归实现
利用非递归的方法实现,也是要按照递归的思路进行的。可能相对来说,有点不是太好理解。
代码如下:
//非递归快排
public void quickSortNo(int[] arr) {
int left = 0;
int right = arr.length-1;
//用栈来记录
Stack<Integer> stack = new Stack<>();
if (left < right) {
stack.push(left);
stack.push(right);
}
while (!stack.isEmpty()) {
//取出栈顶的一对区间值
int right1 = stack.pop();
int left1 = stack.pop();
//分组
int mid = partition3(arr,left1,right1);
//新的分组继续压栈
if (mid-1 > left1) {
stack.push(left1);
stack.push(mid-1);
}
if (mid+1 < right1) {
stack.push(mid+1);
stack.push(right1);
}
}
}
以上就是快速排序的递归、非递归的代码实现过程,个人认为递归的更容易理解。