私はマージソートアルゴリズムを実装するプログラムを作成していますが、毎回2つの部分に分割するのではなく、毎回3つの部分に分割し、再帰的にマージソートします。基本的にはマージソートですが、2つの部分でマージソートするのではなく、毎回3でマージソートするので、かなり楽しいですね。まあ、それは間違いなくそうではありません。
これが私のマージソートの実装です:
public static void mergesort(int[] data) {
int elements = data.length;
int sizeLeft;
int sizeCenter;
int sizeRight;
if (elements > 2) {
if (elements % 3 == 0) {
sizeLeft = elements / 3;
sizeCenter = elements / 3;
sizeRight = elements / 3;
} else if (elements % 3 == 1) {
sizeLeft = (elements / 3) + 1;
sizeCenter = elements / 3;
sizeRight = elements / 3;
} else { //if (elements % 3 == 2)
sizeLeft = (elements / 3) + 1;
sizeCenter = elements / 3;
sizeRight = (elements / 3) + 1;
}
int[] left = makeArray(data, 0, sizeLeft);
int[] center = makeArray(data, sizeLeft, sizeCenter);
int[] right = makeArray(data, sizeLeft + sizeCenter, sizeRight);
mergesort(left);
mergesort(center);
mergesort(right);
merge(data, left, center, right);
}
}
マージ方法は次のとおりです。
public static void merge(int[] data, int[] left, int[] center, int[] right) {
int[] temp = new int[left.length + center.length + right.length];
int copiedTotal = 0;
int copiedLeft = 0;
int copiedCenter = 0;
int copiedRight = 0;
while ((copiedLeft < left.length)
&& (copiedCenter < center.length)
&& (copiedRight < right.length)) {
if ((left[copiedLeft] < center[copiedCenter])
&& (left[copiedLeft] < right[copiedRight])) {
temp[copiedTotal++] = left[(copiedLeft++)];
} else if ((center[copiedCenter] < left[copiedLeft])
&& (center[copiedCenter] < right[copiedRight])) {
temp[copiedTotal++] = center[copiedCenter++];
} else {
temp[copiedTotal++] = right[copiedRight++];
}
}
while ((copiedLeft < left.length) && (copiedCenter < center.length)) {
if (left[copiedLeft] < center[copiedCenter]) {
temp[copiedTotal++] = left[copiedLeft++];
} else{
temp[copiedTotal++] = center[copiedCenter++];
}
}
while ((copiedLeft < left.length) && (copiedRight < right.length)) {
if (left[copiedLeft] < right[copiedRight]) {
temp[copiedTotal++] = left[copiedLeft++];
} else{
temp[copiedTotal++] = right[copiedRight++];
}
}
while ((copiedCenter < center.length) && (copiedRight < right.length)) {
if (center[copiedCenter] < right[copiedRight]) {
temp[copiedTotal++] = center[copiedCenter++];
} else{
temp[copiedTotal++] = right[copiedRight++];
}
}
while (copiedLeft < left.length) {
temp[copiedTotal++] = left[copiedLeft++];
}
while (copiedCenter < center.length) {
temp[copiedTotal++] = center[copiedCenter++];
}
while (copiedRight < right.length) {
temp[copiedTotal++] = right[copiedRight++];
}
System.arraycopy(temp, 0, data, 0, left.length + center.length + right.length);
// for (int i = 0; i < data.length; i++) {
// if ((copiedRight >= right.length) && (copiedCenter >= center.length)) {
// data[i] = left[copiedLeft]; // take from left
// copiedLeft++;
// } else if ((copiedRight >= right.length) && (copiedLeft >= left.length)) {
// data[i] = center[copiedCenter]; // take from left
// copiedCenter++;
// } else if ((copiedCenter >= center.length) && (copiedLeft >= left.length)) {
// data[i] = right[copiedRight]; // take from left
// copiedRight++;
// } else if ((copiedLeft < left.length
// && left[copiedLeft] <= right[copiedRight])
// && left[copiedLeft] <= center[copiedCenter]) {
//
// data[i] = left[copiedLeft]; // take from left
// copiedLeft++;
//
// } else if ((copiedRight >= right.length) && (copiedLeft >= left.length)
// || (copiedCenter < center.length
// && center[copiedCenter] <= right[copiedRight])
// && center[copiedCenter] <= left[copiedLeft]) {
//
// data[i] = center[copiedCenter]; // take from center
// copiedCenter++;
// } else {
// data[i] = right[copiedRight];
// copiedRight++;// take from center
// }
//
// }
}
}
マージメソッド内のコメントには、私が作成しようとした別のマージメソッドがありますが、それはまったくうまく終了せず、事態はさらに複雑になりましたが、参照用にそのままにしておきました。
問題は、これがまったく機能しないことです。たとえば、次のような場合です。
入力:6 5 4 3 2 1
それから私は持っているでしょう:
出力:[2、1、4、3、6、5]
私は正直にこれに一生懸命努力しました、そして2日間続けて、私はこの種のマージソートについて聞いているのは2人だけでした、そしてGoogleで何時間も検索した後、私はここで同様の質問(理解するには複雑すぎました)と別のスレッドを見つけました決して答えられなかったwikiの答えで。
もちろん、私は学ぼうとしているので直接的な解決策を求めているわけではありませんが、ヒントやヒント、そして私が間違ったことをしたことが大いに役立ちます。
前もって感謝します。