13

关于这个问题,答案指定未排序的数组需要更多时间,因为它未能通过分支预测测试。但是如果我们在程序中做一个小的改动:

import java.util.Arrays;
import java.util.Random;


public class Main{

    public static void main(String[] args) {
        // Generate data
        int arraySize = 32768;
        int data[] = new int[arraySize];

        Random rnd = new Random(0);
        for (int c = 0; c < arraySize; ++c) {
            data[c] = rnd.nextInt() % 256;
        }

        // !!! With this, the next loop runs faster
        Arrays.sort(data);

        // Test
        long start = System.nanoTime();
        long sum = 0;

        for (int i = 0; i < 100000; ++i) {
            // Primary loop
            for (int c = 0; c < arraySize; ++c) {
                if (data[c] >= 128) {
                    sum = data[c];
                }
            }
        }

        System.out.println((System.nanoTime() - start) / 1000000000.0);
        System.out.println("sum = " + sum);
    }
}

在这里我已经替换(来自原始问题)

if (data[c] >= 128) 
    sum += data[c];

if (data[c] >= 128) 
    sum = data[c];

未排序的数组给出了大约。同样的结果,我想问为什么分支预测在这种情况下不起作用?

4

1 回答 1

10

我用jmh来分析这个。这是我的代码:

@OutputTimeUnit(TimeUnit.MICROSECONDS)
@BenchmarkMode(Mode.AverageTime)
@Warmup(iterations = 2, time = 1)
@Measurement(iterations = 3, time = 1)
@State(Scope.Thread)
@Fork(2)
public class Comparison
{
  static final int SIZE = 1<<15;
  final int[] data = new int[SIZE];

  @Setup
  public void setup() {
    int i = 1;
    for (int c = 0; c < SIZE; ++c) data[c] = (i*=611953);
    for (int c = 0; c < SIZE; ++c) data[c] = data[c] >= 128? 128 : 127;
  }

  @GenerateMicroBenchmark
  public long sum() {
    long sum = 0;
    for (int c = 0; c < SIZE; ++c) if (data[c] >= 128) sum += data[c];
    return sum;
  }
}

请注意,我不使用排序或随机数生成;它们是不必要的并发症。使用上面代码中使用的公式:

data[c] = (i*=611953);

我得到 132 µs 的运行时间。如果我注释掉涉及的行

data[c] = data[c] >= 128? 128 : 127;

时间完全没有改变。这消除了所有算术考虑并专注于分支预测。如果我使用

data[c] = 127;

我得到 13 µs,如果我使用

data[c] = 128;

我得到 16 µs。这是“基本情况”,强调了不断分支决策之间的差异。

我的结论:这绝对是低级分支预测的效果。

JIT 能否逆转循环?

现在让我们分析您的干预。如果我使用上面代码中提供的公式,但更改

if (data[c] >= 128) sum += data[c];

if (data[c] >= 128) sum = data[c];

那么时间确实从 132 µs 下降到 27 µs。

这是我解释下降的猜测:JIT 编译器可以做的优化技巧是反转循环的方向。现在你的代码变成了

for (int c = SIZE-1; c <= 0; --c) if (data[c] >= 128) { sum = data[c]; break; }

循环已短路到达到与原始循环相同的结果所需的最小迭代次数。

我添加了这个

data[SIZE-1] = 128;

setup()方法结束,但它没有改变时间。这似乎使“循环反转”猜想的幼稚版本无效。

不,应该是cmovl

在分析程序集时,我发现:

cmp edx, 0x80
cmovl eax, ebx

cmovl是一个条件移动指令,它将执行在then分支中发生的赋值的效果,但不涉及任何跳转,因此消除了与分支预测失败相关的任何惩罚。这是对实际效果的一个很好的解释。

于 2014-01-29T14:38:49.760 回答