我正在尝试并行快速排序,因此有一个包含一百万个整数的数组。 有时候会出现以下奇怪的行为:
为了检查数组是否正确排序,我在排序后输入了以下代码:
for(int j=0; j < array_parallel.length-1; ++j)
if(array_parallel[j] > array_parallel[j+1])
System.out.println("ERROR! NOT SORTED CORRECTLY!");
在某些情况下,我会得到错误输出,提示它没有正确排序,当我进行调试时,我会发现以下内容(例如,总是不同):
j=1942 array_parallel[1942] = 6000; array_parallel[1943] = 6000;
(尝试忽略数字,它不是任何特定值或范围)所以只要左边的值等于右边的值,就会出现这种情况。对于大于比较来说,这应该返回false,但我肯定得到了输出。
到底是什么问题呢?
我甚至交叉检查了数组,它的排序是正确的。如果我绘制一个小数组(大约100),也没问题。 我错过了什么吗?
private static int ANZAHL = 1000000; // Größe des Arrays
public static void main(String[] args) {
// TODO Auto-generated method stub
int array_single[] = new int[ANZAHL];
int array_parallel[] = new int[ANZAHL];
Speedmeasure sp_single = new Speedmeasure();
Speedmeasure sp_parallel = new Speedmeasure();
ArrayReader ar = null;
try {
ar = new ArrayReader(array_single, array_parallel);
} catch (FileNotFoundException e1) {
// TODO Auto-generated catch block
e1.printStackTrace();
} catch (IOException e1) {
// TODO Auto-generated catch block
e1.printStackTrace();
} catch (ClassNotFoundException e1) {
// TODO Auto-generated catch block
e1.printStackTrace();
}
if(ar == null) {
System.err.println("Großes Problem. Lass es sein!");
System.exit(-1);
}
else {
for(int i=0; i < 5; ++i) {
Quicksort_Single qs = new Quicksort_Single();
sp_single.setStart(System.currentTimeMillis());
qs.quicksort_start(array_single);
sp_single.setStop(System.currentTimeMillis());
//printArray(array);
PrintSpeed(sp_single.getSpeed(), "Single");
System.out.print("\nUnd jetzt treiben wir es parallel! \n\n");
Thread t1 = new Thread(new Quicksort_Parallel(0, array_parallel.length-1, array_parallel));
sp_parallel.setStart(System.currentTimeMillis());
t1.start();
try {
t1.join();
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
sp_parallel.setStop(System.currentTimeMillis());
//printArray(array_parallel);
PrintSpeed(sp_parallel.getSpeed(),"Parallel");
System.out.println("Speed up was: "+sp_parallel.calcSpeedup(sp_single.getSpeed(), sp_parallel.getSpeed()));
System.out.println("******************************************");
for(int j=0; j < array_single.length-1; ++j)
if(array_single[j] > array_single[j+1])
System.out.println("ERROR! NICHT SORTIERT KORREKT BEI SINGLE!");
for(int j=0; j < array_parallel.length-1; ++j)
if(array_parallel[j] > array_parallel[j+1])
System.out.println("ERROR! NICHT SORTIERT KORREKT BEI PARALLEL!");
ar.copyArray(array_single, array_parallel);
}
}
}
我在启动并行排序的线程上执行了一个join。 第一个线程最多同时生成4个线程。 我不确定它可能是什么并发性,因为我可以在调试器中看到数组已排序。 我将输出两个整数的总和并再次查看。
编辑23/05/12 16:46 UTC+1
我正在更改整个内容以使用JDK 1.7新而真正简单的ForkJoinPool。 对10百万个整数的整数数组进行了测试,并得到了有趣的结果:
我已在Core2Duo(2010)MacBook Pro和Core-i5(2011)Windows 7上进行了测试:
core2duo和i5都可以进行超线程处理,因此现在我尝试使用availableProcessors()* 2-> core2duo加速了一点,以提高2个线程的速度为1.8和1.7; i5目前的加速比为3.2,每个availableProcessors()* 2的线程最多为8个。
仍在对我的计算机进行大量实验。 所有测试都是使用相同的数组进行的,并且从每个数组大小的1000个排序迭代中计算平均值。
>
操作符错误的问题。 发布一个完整的SSCCE以重现您的问题。 - Marko Topolnikt1
的代码中,你是否对它启动的所有4个线程进行了join
?这非常重要。 - Marko Topolnik