我试图在 Java 中实现带有 4 向分区的合并排序算法,问题是它在算法的第 85 行生成了 ArrayIndexOutOfBoundsException
错误。代码如下,我基于Merge Sort
的2-way算法(传统算法):
public static void mergeSort3WayRec(Integer[] gArray, int low, int high,
Integer[] destArray) {
if (high - low < 2) {
return;
}
int mid1 = low + ((high - low) / 4);
int mid2 = low + 2 * ((high - low) / 4) + 1;
int mid3 = low + 3 * ((high - low) / 4) + 2;
mergeSort3WayRec(destArray, low, mid1, gArray);
mergeSort3WayRec(destArray, mid1, mid2, gArray);
mergeSort3WayRec(destArray, mid2, mid3, gArray);
mergeSort3WayRec(destArray, mid3, high, gArray);
merge(destArray, low, mid1, mid2, mid3, high, gArray);
}
public static void merge(Integer[] gArray, int low, int mid1, int mid2, int mid3, int high,
Integer[] destArray) {
int i = low, j = mid1, k = mid2, l = mid3, m = high;
while ((i < mid1) && (j < mid2) && (k < mid3) && (l < high)) {
if (gArray[i].compareTo(gArray[j]) < 0) {
if (gArray[i].compareTo(gArray[k]) < 0) {
if (gArray[i].compareTo(gArray[l]) < 0) {
destArray[m++] = gArray[i++];
} else {
destArray[m++] = gArray[l++];
}
} else {
destArray[m++] = gArray[k++];
}
} else {
if (gArray[j].compareTo(gArray[k]) < 0) {
if (gArray[j].compareTo(gArray[l]) < 0) {
destArray[m++] = gArray[j++];
} else {
destArray[m++] = gArray[l++];
}
} else {
if (gArray[k].compareTo(gArray[l]) < 0) {
destArray[m++] = gArray[k++];
} else {
destArray[m++] = gArray[l++];
}
}
}
}
while ((i < mid1) && (j < mid2)) {
if (gArray[i].compareTo(gArray[j]) < 0) {
destArray[m++] = gArray[i++];
} else {
destArray[m++] = gArray[j++];
}
}
while ((j < mid2) && (k < mid3)) {
if (gArray[j].compareTo(gArray[k]) < 0) {
destArray[m++] = gArray[j++];
} else {
destArray[m++] = gArray[k++];
}
}
while ((k < mid3) && (l < high)) {
if (gArray[k].compareTo(gArray[l]) < 0) {
destArray[m++] = gArray[k++];
} else {
destArray[m++] = gArray[l++];
}
}
while ((i < mid1) && (k < mid3)) {
if (gArray[i].compareTo(gArray[k]) < 0) {
destArray[m++] = gArray[i++];
} else {
destArray[m++] = gArray[k++];
}
}
while ((i < mid1) && (l < high)) {
if (gArray[i].compareTo(gArray[l]) < 0) {
destArray[m++] = gArray[i++];
} else {
destArray[m++] = gArray[l++];
}
}
while ((j < mid2) && (l < high)) {
if (gArray[j].compareTo(gArray[l]) < 0) {
destArray[m++] = gArray[j++];
} else {
destArray[m++] = gArray[l++];
}
}
while (i < mid1) {
destArray[m++] = gArray[i++];
}
while (j < mid2) {
destArray[m++] = gArray[j++];
}
while (k < mid3) {
destArray[m++] = gArray[k++];
}
while (l < high) {
destArray[m++] = gArray[l++];
}
}
需要注意的是,gArray
是main方法中输入的原始数组的副本,这部分代码如下:
public static void main(String args[]) {
Integer[] data = new Integer[]{ 45, -2, -45, 78,
30, -42, 10, 19, 73, 93, 80, 60, 2, 98, 85, 99 };
mergeSort3Way(data);
System.out.println("After 3 way merge sort: ");
for (int i = 0; i < data.length; i++) {
System.out.print(data[i] + " ");
}
}
public static void mergeSort3Way(Integer[] gArray) {
if (gArray == null) {
return;
}
Integer[] fArray = new Integer[gArray.length];
for (int i = 0; i < fArray.length; i++) {
fArray[i] = gArray[i];
}
mergeSort3WayRec(fArray, 0, gArray.length, gArray);
for (int i = 0; i < fArray.length; i++) {
gArray[i] = fArray[i];
}
}
我的问题是,我该如何解决这个错误?另外,如果有额外的实现错误,我已经是做这类算法的新手了。 谢谢。
最佳答案
问题似乎是...,m = high,随后是 destArray[m++] = ...。
在合并中,当 4 路合并到达 4 次运行之一的末尾时,它应该下降到 3 路合并。为了避免重复代码,您需要将索引移动到 low、mid1、mid2,并使用 mid3 或 high 作为从 mid2 开始的子数组的末尾。当 3 向合并到达其中一个运行的末尾时,它应该下降到 2 向合并,然后下降到 1 向复制。
在合并排序中,如果 high-low < 4,您可能只想进行冒泡排序比较并交换 high - low == 3 或 high - low == 2。
假设 high-low < 4 是单独处理的,然后为了稍微均匀地设置内部索引(左边较小的运行):
int mid1 = low +(high+0-low)/4;
int mid2 = mid1+(high+1-low)/4;
int mid3 = mid2+(high+2-low)/4;
自顶向下 4 路合并排序的示例代码使用一对相互递归函数来避免复制回和“展开”合并逻辑。这种方法比做很多条件更快,但我认为主要的性能改进是由于对小运行使用插入排序。在这种情况下,Java 中没有“goto”是一个问题,因为避免重复代码的工作是在合并例程中设置和测试“最小运行”变量。
static final int MINSIZE = 32; // must be >= 3
static void InsertionSort(Integer a[], int ll, int rr)
{
int i = ll+1;
int j;
Integer t;
while(i < rr){
t = a[i];
j = i;
while((j > ll) && a[j-1].compareTo(t)> 0){
a[j] = a[j-1];
j -= 1;}
a[j] = t;
i += 1;}
}
public static void MergeSort(Integer[] a) // entry function
{
if(a.length < 2) // if size < 2 return
return;
Integer[] b = new Integer[a.length];
MergeSortAtoA(a, b, 0, a.length);
}
static void MergeSortAtoA(Integer[] a, Integer[] b, int ll, int rr)
{
if(rr - ll <= MINSIZE){
InsertionSort(a, ll, rr);
return;}
int m1 = ll+(rr+0-ll)/4;
int m2 = m1+(rr+1-ll)/4;
int m3 = m2+(rr+2-ll)/4;
MergeSortAtoB(a, b, ll, m1);
MergeSortAtoB(a, b, m1, m2);
MergeSortAtoB(a, b, m2, m3);
MergeSortAtoB(a, b, m3, rr);
Merge(b, a, ll, m1, m2, m3, rr);
}
static void MergeSortAtoB(Integer[] a, Integer[] b, int ll, int rr)
{
if(rr - ll <= MINSIZE){
System.arraycopy(a, ll, b, ll, rr-ll);
InsertionSort(b, ll, rr);
return;}
int m1 = ll+(rr+0-ll)/4;
int m2 = m1+(rr+1-ll)/4;
int m3 = m2+(rr+2-ll)/4;
MergeSortAtoA(a, b, ll, m1);
MergeSortAtoA(a, b, m1, m2);
MergeSortAtoA(a, b, m2, m3);
MergeSortAtoA(a, b, m3, rr);
Merge(a, b, ll, m1, m2, m3, rr);
}
static void Merge(Integer[] a, Integer[] b, int ll, int m1, int m2, int m3, int rr) {
int bb = ll; // b[] index
int a0 = ll; // a[] indexes
int a1 = m1;
int a2 = m2;
int a3 = m3;
while(true){ // 4 way merge
int sr; // smallest run
if(a[a0].compareTo(a[a1]) <= 0){
if(a[a2].compareTo(a[a3]) <= 0){
if(a[a0].compareTo(a[a2]) <= 0){
sr = 0;}
else{
sr = 2;}}
else{
if(a[a0].compareTo(a[a3]) <= 0){
sr = 0;}
else{
sr = 3;}}}
else{
if(a[a2].compareTo(a[a3]) <= 0){
if(a[a1].compareTo(a[a2]) <= 0){
sr = 1;}
else{
sr = 2;}}
else{
if(a[a1].compareTo(a[a3]) <= 0){
sr = 1;}
else{
sr = 3;}}}
if(sr == 0){
b[bb] = a[a0];
bb++;
a0++;
if(a0 < m1)
continue;
a0 = a1;
a1 = a2;
a2 = a3;
m1 = m2;
m2 = m3;
m3 = rr;
break;}
if(sr == 1){
b[bb] = a[a1];
bb++;
a1++;
if(a1 < m2)
continue;
a1 = a2;
a2 = a3;
m2 = m3;
m3 = rr;
break;}
if(sr == 2){
b[bb] = a[a2];
bb++;
a2++;
if(a2 < m3)
continue;
a2 = a3;
m3 = rr;
break;}
else{ // sr == 3
b[bb] = a[a3];
bb++;
a3++;
if(a3 < rr)
continue;
break;}
}
while(true){ // 3 way merge
int sr; // smallest run
if(a[a0].compareTo(a[a1]) <= 0){
if(a[a0].compareTo(a[a2]) <= 0){
sr = 0;}
else{
sr = 2;}}
else{
if(a[a1].compareTo(a[a2]) <= 0){
sr = 1;}
else{
sr = 2;}}
if(sr == 0){
b[bb] = a[a0];
bb++;
a0++;
if(a0 < m1)
continue;
a0 = a1;
a1 = a2;
m1 = m2;
m2 = m3;
break;}
if(sr == 1){
b[bb] = a[a1];
bb++;
a1++;
if(a1 < m2)
continue;
a1 = a2;
m2 = m3;
break;}
else{ // sr == 2
b[bb] = a[a2];
bb++;
a2++;
if(a2 < m3)
continue;
break;}
}
while(true){ // 2 way merge
if(a[a0].compareTo(a[a1]) <= 0){
b[bb] = a[a0];
bb++;
a0++;
if(a0 < m1)
continue;
a0 = a1;
m1 = m2;
break;}
else{
b[bb] = a[a1];
bb++;
a1++;
if(a1 < m2)
continue;
break;}
}
System.arraycopy(a, a0, b, bb, m1-a0); // 1 way copy
}
修复了 chqrlie 的代码。
public static void merge(Integer[] gArray, int low, int mid1, int mid2, int mid3, int high,
Integer[] destArray)
{
int i = low, j = mid1, k = mid2, l = mid3, m = low;
while (m < high) {
if (i < mid1 && (j >= mid2 || gArray[i].compareTo(gArray[j]) <= 0)) {
if (k >= mid3 || gArray[i].compareTo(gArray[k]) <= 0) {
if (l >= high || gArray[i].compareTo(gArray[l]) <= 0) {
destArray[m++] = gArray[i++];
} else {
destArray[m++] = gArray[l++];
}
} else {
if (k < mid3 && (l >= high || gArray[k].compareTo(gArray[l]) <= 0)) {
destArray[m++] = gArray[k++];
} else {
destArray[m++] = gArray[l++];
}
}
} else {
if (j < mid2 && (k >= mid3 || gArray[j].compareTo(gArray[k]) < 0)) {
if (l >= high || gArray[j].compareTo(gArray[l]) < 0) {
destArray[m++] = gArray[j++];
} else {
destArray[m++] = gArray[l++];
}
} else {
if (k < mid3 && (l >= high || gArray[k].compareTo(gArray[l]) < 0)) {
destArray[m++] = gArray[k++];
} else {
destArray[m++] = gArray[l++];
}
}
}
}
}
public static void mergeSort4WayRec(Integer[] gArray, int low, int high,
Integer[] tempArray) {
if (high - low < 2) {
return;
}
int mid1 = low + (high + 0 - low) / 4;
int mid2 = mid1 + (high + 1 - low) / 4;
int mid3 = mid2 + (high + 2 - low) / 4;
mergeSort4WayRec(tempArray, low, mid1, gArray);
mergeSort4WayRec(tempArray, mid1, mid2, gArray);
mergeSort4WayRec(tempArray, mid2, mid3, gArray);
mergeSort4WayRec(tempArray, mid3, high, gArray);
merge(tempArray, low, mid1, mid2, mid3, high, gArray);
}
public static void mergeSort4Way(Integer[] gArray) {
if (gArray != null) {
Integer[] tempArray = new Integer[gArray.length];
for (int i = 0; i < gArray.length; i++) {
tempArray[i] = gArray[i];
}
mergeSort4WayRec(gArray, 0, gArray.length, tempArray);
}
}
public static void main(String[] args) {
Integer[] a = new Integer[1024*1024];
Random r = new Random();
for(int i = 0; i < a.length; i++)
a[i] = r.nextInt();
long bgn, end;
bgn = System.currentTimeMillis();
mergeSort4Way(a);
end = System.currentTimeMillis();
for(int i = 1; i < a.length; i++){
if(a[i-1] > a[i]){
System.out.println("failed");
break;
}
}
System.out.println("milliseconds " + (end-bgn));
}
关于java - 如何在没有错误 ArrayIndexOutOfBoundsException 的情况下实现具有 4 向分区的合并排序算法?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55840901/