
快速排序是一种高效的、基于比较的排序算法,采用分治(divide and conquer)策略。其核心思想是:从数组中选择一个元素作为“枢轴”(pivot),然后将数组分为两部分,使得所有小于枢轴的元素都位于枢轴之前,所有大于枢轴的元素都位于枢轴之后。这个过程称为“分区”(partition)。分区完成后,枢轴就处于其最终的排序位置。接着,对枢轴左右两边的子数组递归地重复这个过程,直到所有元素都排好序。
在实现递归快速排序时,开发者常遇到一些微妙的错误,导致排序结果不正确或出现栈溢出。这些问题通常集中在以下几个方面:
下面我们将通过一个修正后的Java实现来详细说明如何解决上述问题,构建一个健壮的快速排序算法。
这个方法是公共接口,负责调用实际的递归排序方法。
public class QuickSort {
public static void quickSort(int[] s) {
if (s == null || s.length < 2) { // 处理空数组或单元素数组的边界情况
return;
}
quickSortSub(s, 0, s.length - 1);
}
// ... 其他方法
}这是快速排序的核心递归逻辑。
立即学习“Java免费学习笔记(深入)”;
private static void quickSortSub(int[] s, int a, int b) {
// 递归基准条件:当子数组至少包含两个元素时才进行分区和递归
// b - a >= 1 表示子数组长度至少为2 (b - a + 1 >= 2)
if (b - a >= 1) {
int point = partition(s, a, b); // 执行分区操作,获取枢轴的最终位置
// 递归排序左子数组
// 只有当左子数组至少包含两个元素时才递归,避免对单元素或空数组进行不必要的递归调用
if (point > a) { // point > a 意味着左子数组至少有一个元素
quickSortSub(s, a, point - 1);
}
// 递归排序右子数组
// 只有当右子数组至少包含两个元素时才递归
if (point < b) { // point < b 意味着右子数组至少有一个元素
quickSortSub(s, point + 1, b);
}
}
}关键修正点说明:
分区方法是快速排序算法的灵魂,负责将数组分为两部分并放置枢轴。
private static int partition(int[] s, int a, int b) {
int pivot = s[b]; // 选择最右边的元素作为枢轴
int left = a; // 左指针,从子数组起始位置开始
int right = b - 1; // 右指针,从枢轴左边一个位置开始
while (left <= right) { // 循环直到左右指针相遇或交叉
// 移动左指针,直到找到一个大于或等于枢轴的元素
while (left <= right && s[left] < pivot) {
left++;
}
// 移动右指针,直到找到一个小于或等于枢轴的元素
while (left <= right && s[right] > pivot) {
right--;
}
// 如果左右指针尚未相遇或交叉,则交换它们指向的元素
if (left <= right) {
int tmp = s[left];
s[left] = s[right];
s[right] = tmp;
// 交换后,指针继续向内移动
left++;
right--;
}
}
// 循环结束后,left 指针指向第一个大于或等于枢轴的元素的位置
// 将枢轴放到其最终位置
int tmp = s[left];
s[left] = s[b]; // 将枢轴(s[b])与 s[left] 交换
s[b] = tmp;
return left; // 返回枢轴的最终位置
}关键修正点说明:
public class QuickSort {
public static void quickSort(int[] s) {
if (s == null || s.length < 2) {
return;
}
quickSortSub(s, 0, s.length - 1);
}
private static void quickSortSub(int[] s, int a, int b) {
if (b - a >= 1) { // 确保子数组至少包含两个元素
int point = partition(s, a, b); // 执行分区操作
// 递归排序左子数组
if (point > a) { // 确保左子数组不为空
quickSortSub(s, a, point - 1);
}
// 递归排序右子数组
if (point < b) { // 确保右子数组不为空
quickSortSub(s, point + 1, b);
}
}
}
private static int partition(int[] s, int a, int b) {
int pivot = s[b]; // 选择最右边的元素作为枢轴
int left = a; // 左指针
int right = b - 1; // 右指针
while (left <= right) { // 循环直到左右指针相遇或交叉
// 移动左指针
while (left <= right && s[left] < pivot) {
left++;
}
// 移动右指针
while (left <= right && s[right] > pivot) {
right--;
}
// 如果指针未交叉,则交换元素
if (left <= right) {
int tmp = s[left];
s[left] = s[right];
s[right] = tmp;
left++;
right--;
}
}
// 将枢轴归位
int tmp = s[left];
s[left] = s[b];
s[b] = tmp;
return left; // 返回枢轴的最终位置
}
public static void main(String[] args) {
int[] arr = {85, 10, 24, 63, 45, 27, 100, 31, 96, 50, 40, 23, 49, 96, 120, 105, 13, 5, 42, 69, 22, 12};
System.out.println("原始数组:");
for (int i : arr) System.out.print(i + ", ");
System.out.println("\n");
quickSort(arr);
System.out.println("排序后数组:");
for (int i : arr) System.out.print(i + ", ");
System.out.println("");
}
}测试输出:
原始数组: 85, 10, 24, 63, 45, 27, 100, 31, 96, 50, 40, 23, 49, 96, 120, 105, 13, 5, 42, 69, 22, 12, 排序后数组: 5, 10, 12, 13, 22, 23, 24, 27, 31, 40, 42, 45, 49, 50, 63, 69, 85, 96, 96, 100, 105, 120,
可以看到,经过修正后的代码能够正确地对数组进行排序。
正确实现快速排序需要对递归基准条件和分区逻辑有深入的理解。以下是一些关键点:
通过遵循这些最佳实践,可以构建出高效且鲁棒的快速排序算法。
以上就是深入理解与修正:Java递归实现快速排序的常见陷阱与最佳实践的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号