我的快速排序方法未按预期工作

My quicksort method is not working as intended

提问人:edgeworth miles 提问时间:11/17/2023 最后编辑:user207421edgeworth miles 更新时间:11/18/2023 访问量:59

问:

我尝试创建一个快速排序方法,其唯一参数是 MyVector'(这只是一个普通向量),但该方法没有按预期工作,我不确定为什么。

预期输出应该是

The element at index 0 is: 7
The element at index 0 is: 10
The element at index 0 is: 28

但我得到的是

The element at index 0 is: 41524
The element at index 1 is: 26432
The element at index 2 is: 51244

驱动程序类 (示例)

    MySort.QuickSort(testing);
    System.out.println("The element at index 0 is: "+testing.elementAt(0));
    System.out.println("The element at index 1 is: "+testing.elementAt(1));
    System.out.println("The element at index 2 is: "+testing.elementAt(2));

MySort 类

public static void QuickSort(MyVector vec){
    MyVector lo = new MyVector();
    MyVector hi = new MyVector();
    int loSize = lo.size();
    int hiSize = hi.size();
    if (loSize >= hiSize)
        return;
    int p = (int)vec.elementAt((loSize + hiSize) / 2);   // set pivot, could use median of 3 here
    int i = loSize-1;
    int j = hiSize+1;
    while (true){
        while ((int)vec.elementAt(i) < p) {
        ++i;
        }
        while ((int)vec.elementAt(j) > p) { ;    // decrease j until a[j] <= pivot
        --j;
        }
        if (i >= j)             // break if indices meet or cross
            break;
        swap((int)vec.elementAt(i), (int)vec.elementAt(j));
    }
    QuickSort(lo);        
    QuickSort(hi); 
      }
public static void swap(int a, int b) {
    int temp;
    temp = a;
    a=b;
    b=temp;
}
Java 向量 快速排序

评论

1赞 MadProgrammer 11/17/2023
我不认为你的交换会起作用,因为你实际上并没有改变中包含的值(除非你需要同步访问,否则最好使用某种)。VectorList
0赞 edgeworth miles 11/17/2023
@MadProgrammer 可悲的是,我不能只换一个列表。此分配需要使用向量作为先决条件
0赞 MadProgrammer 11/17/2023
那么问题仍然存在,您需要交换列表/向量中的元素,而不是值
0赞 edgeworth miles 11/17/2023
我只是更改了行以交换元素,因为它不起作用 swap((int)vec.elementAt(i), (int)vec.elementAt(j));换成 swap(vec.elementAt(i), vec.elementAt(j));并将交换方法转换为公共静态 void swap(Object object, Object object2) { Object temp; temp = object; object=object2; object2=temp; }
0赞 MadProgrammer 11/17/2023
另一个问题是 / 概念,它们应该表示索引范围(或您要排序的范围),并且不应该是新的向量。然后,枢轴将是这两者之间的中间品脱,然后作为逻辑结果的比较值hilo

答:

1赞 MadProgrammer 11/17/2023 #1

首先,这个...

swap((int)vec.elementAt(i), (int)vec.elementAt(j));
//...
public static void swap(int a, int b) {
    int temp;
    temp = a;
    a=b;
    b=temp;
}

是错误的。你交换了值,但你从不交换 中的元素,并且由于 Java 的工作方式,一旦方法存在,交换就会丢失Vector

相反,它应该更像是......

protected void swap(MyVector vector, int index1, int index2) {
    Integer value1 = vector.get(index1);
    Integer value2 = vector.get(index2);
    vector.set(index1, value2);
    vector.set(index2, value1);
}

接下来,这个...

MyVector lo = new MyVector();
MyVector hi = new MyVector();
int loSize = lo.size();
int hiSize = hi.size();
if (loSize >= hiSize)
    return;
int p = (int)vec.elementAt((loSize + hiSize) / 2);

没有任何意义。为什么要创建两个新的(我猜)空向量?这些将如何帮助你??

和值应将范围括在要搜索的范围内(即,从索引到索引)。lo(w)hi(gh)Vectorlohi

then 表示要用于确定如何最好地执行排序的核心逻辑的“基本”比较值。p(ivot)

我会重写它,但老实说,逻辑是错误的。你应该从一个只接受 的方法开始,然后从中调用一个单独的方法,该方法接受 AND the (默认为 ) 和 (默认为 ) 值 - 这构成了递归的基础。sortVectorsortVectorlo0hiVector#size - 1

这意味着,当你完成一个传递时,你就会确定递归的下一阶段,比如......

if (low < j) {
    sort(vector, low, j);
}
if (i < high) {
    sort(vector, i, high);
}

现在,对我来说,我将从一个简单的类开始,它可以封装这个工作流并“隐藏”一些实现细节。你不需要“必须”这样做,但它会让你的一些工作更容易,因为你不需要不断地传递。Vector

下面的例子基本上创建一个 ,用一些整数填充它并随机化它。Vector

然后,它会对值进行排序并打印值。您应该注意的是,这不会进行“就地”替换,也就是说,排序将创建第一个副本的副本,然后对副本进行排序,将其返回给调用方。Vector

import java.util.Collections;
import java.util.Random;
import java.util.Vector;

public class Main {
    public static void main(String[] args) {
        Random rnd = new Random();
        Vector<Integer> master = new Vector<>(8);
        for (int index = 0; index < 8; index++) {
            master.add(index);
        }
        Collections.shuffle(master);

        Vector<Integer> sorted = QuickSort.sort(master);
        System.out.println("Unsorted = " + master);
        System.out.println("  Sorted = " + sorted);
    }

    public static class QuickSort {
        private Vector<Integer> master;

        private QuickSort(Vector<Integer> master) {
            // Make a shallow copy
            this.master = new Vector<>(master);
        }

        public static Vector<Integer> sort(Vector<Integer> master) {
            QuickSort quickSort = new QuickSort(master);
            return quickSort.sort();
        }

        protected Vector<Integer> sort() {
            sort(0, master.size() - 1);
            return master;
        }

        protected void sort(int low, int high) {
            int i = low, j = high;

            // Get the pivot element
            int pivot = low + (high - low) / 2;
            Integer value = master.get(pivot);

            System.out.println("low = " + low + "; high = " + high);
            // Divide into two lists
            while (i <= j) {

                while (master.get(i) < value) {
                    i++;
                }

                while (master.get(j) > value) {
                    j--;
                }

                if (i <= j) {
                    System.out.println("Swap " + master.get(i) + " <> " + master.get(j));
                    swap(i, j);
                    i++;
                    j--;
                }
            }

            // Recursion
            if (low < j) {
                sort(low, j);
            }
            if (i < high) {
                sort(i, high);
            }
        }

        protected void swap(int index1, int index2) {
            Integer value1 = master.get(index1);
            Integer value2 = master.get(index2);
            master.set(index1, value2);
            master.set(index2, value1);
        }
    }
}

我会考虑看看:

有关快速排序通常如何工作的更多详细信息