【问题标题】:Java Integer Comparison: greater thanJava整数比较:大于
【发布时间】:2012-05-29 08:09:21
【问题描述】:

我有一个包含一百万个整数的数组,因为我正在尝试并行快速排序。 我有时会出现以下奇怪的行为:

为了检查数组是否正确排序,我在排序后输入了以下代码:

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 个)也可以。 我是不是错过了什么让我心烦意乱的事情?

21:32 (UTC+1) 编辑:

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);
            }
        }
    }

我在启动并行排序的线程上进行连接。第一个线程最多同时产生最多 4 个线程。我不能 100% 确定它可能是什么并发,正如我在调试器中看到的那样,数组已排序。我将两个整数的输出相加,再看看。

于 2012 年 5 月 23 日 16:46 UTC+1 编辑

我正在改变整个事情以使用来自 JDK 1.7 的新的、非常简单的 ForkJoinPool。 使用最多 10 个 mio 整数的整数数组进行测试,得到了有趣的结果: 我已经在 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 次排序迭代计算得出的。

【问题讨论】:

  • 您是否在多个线程中运行排序算法?
  • 为什么不发布生成输出的确切代码。这肯定不是&gt; 操作员错了。发布完整的SSCCE 以重现您的问题。
  • 很难说没有看到其余代码,但这可能是由于并发问题,您在排序算法期间所做的某些更改对检查结果的线程不可见。跨度>
  • 刚刚完成工作的线程在没有适当同步的情况下不能保证任何事情。线程可能还活着,并且有一些线程本地存储尚未提交到主内存。
  • 是的,在线程t1 的代码中,你在它启动的所有4 个线程上都join 吗?这很重要。

标签: java comparison integer


【解决方案1】:

查看您的代码,您生成了一个线程,然后立即将其加入主执行线程:

        Thread t1 = new Thread(new Quicksort_Parallel(0, array_parallel.length-1, array_parallel));

        sp_parallel.setStart(System.currentTimeMillis());
        t1.start();

        try {
            t1.join();

问题变成了 - 您在 Quicksort_Parallel 例程中做什么?你是否产生了额外的线程?您是否正在加入所有?如果不是,您已经创建了一个竞争条件来解释您所看到的结果。

【讨论】:

  • 正如我在上面的 cmets 中提到的,我的 quicksort_parallel 最多可能产生 3 个其他线程。我没有参与其中 - 正如我们所说,这是一个竞争条件。如果我加入 quicksort_parallel 我的排序比“正常”快速排序慢得多。
  • 与生成线程相关的开销。要执行的工作是否足够大以受益于并行性?
  • 我测试了多达 1000 万个整数。
【解决方案2】:

您可能是竞争条件的受害者,这意味着您的输出取决于其他事件的顺序。因此,如果您有一个工作速度更快的线程,您将获得竞争条件。阻止这种情况的一种方法是使用信号量或在线程之间划分循环。你怎么样?你在使用减法吗?

【讨论】:

    【解决方案3】:

    【讨论】:

    • 感谢您的提示 - 我今天在另一个版本中实现了 ForkJoinPool,对于 1 个 mio 整数,平均加速约为 1.5。将生成一个包含多达 100 个 mio 整数的大型测试并进行测试。但是 1.5 仍然不多,也不是我所期待和希望的
    • 它的 Intel Core2Duo (2010)。将在家里对 2011 年末的 Core-i5 进行交叉检查
    • 我猜如果线程多于内核,性能会下降,因为内核必须在线程之间切换。
    • 这是一个逻辑链接,但也涉及到有关 cpu 的知识。 core2duo 和 i5 可以进行超线程,所以我现在使用 availableProcessors()*2 进行了测试 -> core2duo 对于 10 个 mio 整数得到了一点提升,最高可达 1.8; 1.7 用于 2 个线程; i5 目前的加速比约为 3.2,每个 availableProcessors()*2 最多 8 个线程。我想我在这里向一些非常好的提示和问题解决者证明了我想要的东西!
    猜你喜欢
    • 2014-11-07
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2018-04-07
    • 2013-02-27
    • 1970-01-01
    • 1970-01-01
    • 2011-12-29
    相关资源
    最近更新 更多