快速排序越界或循环

Quicksort out of bounds or loop

提问人:Veee 提问时间:11/14/2023 更新时间:11/14/2023 访问量:33

问:

我目前正在尝试在 java 中实现快速排序,当我运行程序时,我要么得到数组长度的索引越界,要么得到无限循环,不要问我怎么做。此外,交换本身在工作时是不正确的,所以是的。

public class Quicksort {

   /*
    * Entry method for Quicksort
    */
   public static void quicksort(int[] a) {
      quicksort(a, 0, a.length-1);
   }

   /*
    * Quicksort! 
    * Operates in-place, i.e. it doesn't create a copy.
    */  
   public static void quicksort(int[] a, int from, int to) {
      // Base case: Sorting range is 1 element or empty.
      if(from >= to){
         return;
      }
      // TODO
      else{
         
         
         
         int ipivot = from; // Pivot p
         System.out.println("Pivot p = " + a[ipivot]);
         printArray(a);
         
         // Split the array in two ranges: Left are the elements <=p and right >p.
         // Do this by swapping the elements that don't belong in the given range. 
         // If i and j cross, swapping is finished.
         int i = ipivot+1;
         int j = to;
         while (i<j) {
            // Look for an element <=p right-hand side 
            while(j>= 0 &&a[j]>a[ipivot]){
               j--;
            }
            
            
            // Look for an element >p left-hand side 
            while(i< a.length && a[i]<a[ipivot]){
               i++;
            }
            
            
            // Swap the elements, but only if pointers aren't crossed.
            if (i<j) {
               swap(a, i, j);
            } 
            else {
               break;
            }
 
               
         }
         // The pivot can now be moved to its final position at i.
         // NOTE: Check if pivot needs to be moved BEFORE i.
         int ipivot_final =i;
         System.out.println("swapping " + a[ipivot] +" @"+ipivot+" and " + a[ipivot_final]+" @"+ipivot_final + " (pivot)");
         swap(a, ipivot, ipivot_final);
         
         printArray(a);
         System.out.println();
         
         // Recursion case: Sort left and right range of pivot recursively.
         quicksort(a, from, ipivot_final-1);
         quicksort(a, ipivot_final+1, to);
      }
   }
   /*
    * HELPER METHODS
    */
   public static void printArray(int[] a) {
      for (int i=0; i<a.length; i++) {
         System.out.print(a[i] + ", ");
      }
      System.out.println();
   }

   public static boolean isSorted( int[] array ) {
      for ( int j=0; j<array.length-1; j++ ) {
         if (array[j] > array[j+1])
            return false;      
      }
      return true;
   }

   private static int[] createRandomArray(int size, int minVal, int maxVal) {
      int[] a = new int[size];
      for (int i=0; i<a.length; i++) {
         a[i] = (int) (Math.random()*(maxVal-minVal) + minVal);
      }
      return a;
   }

   private static void swap(int[] a, int i, int j) {
      int tmp = a[i];
      a[i] = a[j];
      a[j] = tmp;
   }
   
   
   /*
    * MAIN
    */

   public static void main(String[] args) {
      final int MINSIZE = 10;
      final int MAXSIZE = 10;
      for (int size = MINSIZE; size <= MAXSIZE; size++) {
         int[] a = createRandomArray(size, 0, size);
         System.out.println("Unsorted array of size "+a.length);
         printArray(a);
         System.out.println("\nRunning Quicksort ...");
         quicksort(a);
         System.out.println("\nSorted array:");
         printArray(a);
         assert isSorted(a) : "Array is not sorted!";
         if (isSorted(a))
            System.out.println("\n\nSuccess!!\n\n");
         else
            System.out.println("\n\nERROR !! Something went wrong.\n\n");
      }
   }

}

我尝试更改条件的顺序以及条件本身

Java 快速排序

评论


答:

0赞 John Bollinger 11/14/2023 #1

这些循环中的条件是错误的:

            while(j>= 0 &&a[j]>a[ipivot]){
               j--;
            }
            
            
            // Look for an element >p left-hand side 
            while(i< a.length && a[i]<a[ipivot]){
               i++;
            }

j不该被允许走得更远,也不应该被允许走在上面。或者,您可以尽快断开循环并交叉:fromitoij

            while (j >= i && a[j] > a[ipivot]) {
               j--;
            }
            
            
            // Look for an element >p left-hand side 
            while (i <= j && a[i] < a[ipivot]) {
               i++;
            }

此评论不正确:

         // The pivot can now be moved to its final position at i.

枢轴的最终位置不是 ,而是 。这也可以拼写,因为循环会在交叉时立即终止。ii-1jij

一种方法是考虑极端情况,即枢轴元素恰好是子数组中最少的。则永远不会从其初始值 递增,但枢轴元素的正确索引是其初始索引 。iipivot + 1ipivot

另请注意,如果您允许一直增加到此时,评估将产生 .这也与以下事实有关:在分区循环结束时,左侧分区的顶部索引是 ,而不是 。ia.lengtha[i]IndexOutOfBoundsExceptioni - 1i

0赞 Reilas 11/14/2023 #2

这是一个实现,使用维基百科文章中的算法。
维基百科 – 快速排序 – Lomuto 分区方案

评论来自原文。

// Sorts a (portion of an) array, divides it into partitions, then sorts those
void quicksort(int[] A, int lo, int hi) {
    // Ensure indices are in correct order
    if (lo >= hi || lo < 0) return;

    // Partition array and get the pivot index
    int p = partition(A, lo, hi);

    // Sort the two partitions
    quicksort(A, lo, p - 1); // Left side of pivot
    quicksort(A, p + 1, hi); // Right side of pivot
}

// Divides array into two partitions
int partition(int[] A, int lo, int hi) {
    int pivot = A[hi]; // Choose the last element as the pivot

    // Temporary pivot index
    int i = lo - 1, t;

    for (int j = lo; j <= hi - 1; j++) {
        // If the current element is less than or equal to the pivot
        if (A[j] <= pivot) {
            // Move the temporary pivot index forward
            i = i + 1;
            // Swap the current element with the element at the temporary pivot index
            t = A[i];
            A[i] = A[j];
            A[j] = t;
        }
    }

    // Move the pivot element to the correct pivot position (between the smaller and larger elements)
    i = i + 1;
    t = A[i];
    A[i] = A[hi];
    A[hi] = t;
    return i; // the pivot index
}