0

我正在使用 Medians 枢轴方法的 Median 实现一个 select-kth 算法。具体来说,我正在遵循此处列出的伪代码。. 但是,我的代码崩溃(下面讨论的错误),我明白它为什么会崩溃,但我不明白我能做些什么。

错误

Exception in thread "main" java.lang.ArrayIndexOutOfBoundsException: 32017219
at Selection.partition(Selection.java:173)

原因
在 wiki 链接中,该方法select将返回一个if left == right。但是,当从 的 return 语句调用相互递归时pivot,这意味着它将一个值(而不是索引)返回给父级,并且父级将该值设置为pivotIndex.

编码

public static int alg5(int[] a, int left, int right, int k) {
    if (left == right) {
        return a[left];
    }

    while (true) {
        int pivotIndex = pivot(a, left, right);         
        pivotIndex = partition(a, left, right, pivotIndex);

        if (k == pivotIndex) {
            return a[k];
        } else if (k < pivotIndex) {
            right = pivotIndex - 1;
        } else {
            left = pivotIndex + 1;
        }

    }
}

public static int pivot(int[] a, int left, int right) {
    for (int i = left; i <= right; i = i + 5) {
        int subRight = i + 4;
        if (subRight > right) {
            subRight = right;
        }

        int median5 = partition5(a, i, subRight);
        swap(a, median5, (int)(Math.floor(i / 5)));
    }

    return alg5(a, left, (int)(left + Math.ceil((right - left) / 5) - 1), ((right - left) / 10));
}

public static int partition5(int[] a, int left, int right) {
    Arrays.sort(a);

    return ((a.length - 1) / 2);
}

public static int partition(int[] a, int left, int right, int pivotIndex) {
    int pivotValue = a[pivotIndex];
    a = swap(a, pivotIndex, right);
    int storeIndex = left;

    for (int i = left; i <= right; i++) {
        int aOfi = a[i];
        if (a[i] < pivotValue) {
            swap(a, storeIndex, i);
            storeIndex++;
        }
    }

    swap(a, right, storeIndex);

    return storeIndex;
}

我完全理解为什么我的代码不起作用,我只是不明白如何修复它,因为它看起来正是算法指定的。非常感谢指针!

4

1 回答 1

1

有很多错误:

  1. 该方法pivot不应修改数组a。它应该只是为未来找到一个支点partition。一个肮脏的解决方法是调用pivot(a.clone(), left, right);. (你不应该这样做,只是为了给你一个想法。)

  2. Math.ceil((right - left) / 5)是整数除法。您应该将它们转换为浮点数:Math.ceil(((float)(right - left)) / 5f).

  3. partition5中,您对整个数组进行排序a!您应该只进行比较以找到和之间的中位数的索引a[left]a[right]

  4. 在某些时候你可能有right < left,所以你的第一行alg5应该写if (left >= right)

于 2015-05-26T21:53:38.133 回答