0

所以,我写了一个快速排序算法和一个 hoare-partition 算法。不知何故,当我尝试在 main() 中运行示例案例时,它挂断了 quickSort(test, 0,3)。似乎有一个无限循环。我不知道如何解决它,因为这两个功能似乎都很好。

我尝试调试,但我对 c 相当陌生。我注意到 quickSort(test,0,3) 递归调用自己。所以我知道这个问题与高而不是减少有关。但是我从大学幻灯片中获取了示例伪代码来构建该功能,并且一切似乎都排成一行。

void printArray(int A[], int high) {
    for (int i=0; i<=high; i++) {
         printf("%d ", A[i]);
    }
}

int partitionHoare(int A[], int low, int high) {
    int pivot=A[low];
    int left = low-1;
    int right= high+1;

    while (1){
        while (1){
            left++;
            if (A[left]>=pivot) break;
        }
        while (1){
            right--;
            if (A[right]<=pivot) break;
        }


        if (left<right)  {
            int temp=A[left];
            A[left]=A[right];
            A[right]=temp;
        }
        else return left;
    }
}

void quicksort(int A[], int low, int high){
    if (low<high){
        int middle=partitionHoare(A,low, high);
        quicksort(A, low,middle-1);
        quicksort(A, middle, high);
    }
}


void main (){
    int test[]={64,81,24,42,90,30,9,95};
    quicksort(test,0,7);
    printArray(test,7);

我实际上希望打印出的测试数组是这样排序的:“9, 24, 30, 42, 64, 81, 90, 95”

4

2 回答 2

1

您的quicksort()功能存在逻辑缺陷:

void quicksort(int A[], int low, int high){
    if (low<high){
        int middle=partitionHoare(A,low, high);
        quicksort(A, low,middle-1);
        quicksort(A, middle, high);
    }
}

它不确保递归将终止。

具体来说,如果在某个长度大于 1 的子数组中,第一个元素是最少的并且不是重复的,那么partitionHoare()将返回一个等于low而不修改数组的值。在这种情况下,对左子数组的递归调用将什么都不做,但对子数组的递归调用将准确地重复当前参数。什么都没有改变,同样的事情肯定会再次发生,一次又一次,无限期地发生。

quicksort()在这种情况下,您可以通过在if中进行测试来打破无限递归middle == low,但这并不能为您提供正确的排序。

这里的一种常见解决方案是双重的:

  1. 确保分区函数将枢轴值交换为其报告的枢轴索引。这肯定是该值的正确最终位置。
  2. 递归时,从两个子数组中排除枢轴索引(我们知道它的值是正确的),这样每个子问题肯定小于父问题。
于 2019-04-25T15:40:54.640 回答
0

更改分区以返回右(而不是左)。将两个递归调用更改为 | 快速排序(A,低,中);| 快速排序(A,中间+1,高);。这将与 wiki 文章的示例完全匹配:

https://en.wikipedia.org/wiki/Quicksort#Hoare_partition_scheme

您可能想要更改“中间”的名称。Wiki 示例将其称为 p 表示分区拆分索引(而不是表示到枢轴的索引),因为使用 Hoare 分区方案,枢轴和等于枢轴的元素可能会在左分区的末尾和/或在正确的分区。

于 2019-04-26T01:23:13.303 回答