gpt4 book ai didi

java - 多线程合并排序,添加额外的线程

转载 作者:塔克拉玛干 更新时间:2023-11-02 08:40:14 25 4
gpt4 key购买 nike

我在 java 中的多线程归并排序算法中遇到了一个问题。

我应该将代码修改为 3、4、5、6、7、8 线程合并排序,方法是将原始数组划分为 subArray。目前它有 2 个 subArray。如何将原始数组拆分为 3、4、5、6、7、8 个 subArray 以实现我的目标?此外,我应该再写一些方法,因为 mergeSort 方法调用了 lefthalfrighthalf 方法。所以对于 3、4、5、6、7、8 个线程,我应该编写额外的方法。我该如何处理?

two_threaded_merge_sort.java

public class two_threaded_merge_sort {

public static void finalMerge(int[] a, int[] b) {
int[] result = new int[a.length + b.length];
int i=0;
int j=0;
int r=0;
while (i < a.length && j < b.length) {
if (a[i] <= b[j]) {
result[r]=a[i];
i++;
r++;
} else {
result[r]=b[j];
j++;
r++;
}
if (i==a.length) {
while (j<b.length) {
result[r]=b[j];
r++;
j++;
}
}
if (j==b.length) {
while (i<a.length) {
result[r]=a[i];
r++;
i++;
}
}
}
}

public static void main(String[] args) throws InterruptedException {
Random rand = new Random();
int[] original = new int[9000000];
for (int i=0; i<original.length; i++) {
original[i] = rand.nextInt(1000);
}

long startTime = System.currentTimeMillis();
int[] subArr1 = new int[original.length/2];
int[] subArr2 = new int[original.length - original.length/2];
System.arraycopy(original, 0, subArr1, 0, original.length/2);
System.arraycopy(original, original.length/2, subArr2, 0, original.length - original.length/2);

Worker runner1 = new Worker(subArr1);
Worker runner2 = new Worker(subArr2);
runner1.start();
runner2.start();
runner1.join();
runner2.join();
finalMerge (runner1.getInternal(), runner2.getInternal());
long stopTime = System.currentTimeMillis();
long elapsedTime = stopTime - startTime;
System.out.println("2-thread MergeSort takes: " + (float)elapsedTime/1000 + " seconds");
}

}

worker .java

class Worker extends Thread {
private int[] internal;

public int[] getInternal() {
return internal;
}

public void mergeSort(int[] array) {
if (array.length > 1) {
int[] left = leftHalf(array);
int[] right = rightHalf(array);

mergeSort(left);
mergeSort(right);

merge(array, left, right);
}
}

public int[] leftHalf(int[] array) {
int size1 = array.length / 2;
int[] left = new int[size1];
for (int i = 0; i < size1; i++) {
left[i] = array[i];
}
return left;
}

public int[] rightHalf(int[] array) {
int size1 = array.length / 2;
int size2 = array.length - size1;
int[] right = new int[size2];
for (int i = 0; i < size2; i++) {
right[i] = array[i + size1];
}
return right;
}

public void merge(int[] result, int[] left, int[] right) {
int i1 = 0;
int i2 = 0;

for (int i = 0; i < result.length; i++) {
if (i2 >= right.length || (i1 < left.length && left[i1] <= right[i2])) {
result[i] = left[i1];
i1++;
} else {
result[i] = right[i2];
i2++;
}
}
}

Worker(int[] arr) {
internal = arr;
}

public void run() {
mergeSort(internal);
}
}

非常感谢!

最佳答案

需要有一个排序函数将数组分成 k 个部分,然后创建 k 个线程对每个部分进行排序,使用自上而下或自下而上的方法,(自下而上会稍微快一些),并等待所有线程完成完成。

此时有k个排序好的部分。这些可以使用 k 路合并(复杂)一次合并,或者一次合并一对部分(2 路合并),也许使用多个线程,但此时该过程可能是内存带宽有限,所以多线程可能帮不上什么忙。

当将数组分成 k 个部分时,可以使用类似这样的方法来保持大小相似:

int r = n % k;
int s = n / k;
int t;
for each part{
t = r ? 1 : 0;
r -= t;
size = s + t;
}

int r = n % k;
int s = n / k + 1;
while(r--){
next part size = s; // n / k + 1
}
s -= 1;
while not done{
next part size = s; // n / k
}

关于java - 多线程合并排序,添加额外的线程,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34580668/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com