3

我正在处理 Java 中的多线程,正如有人向我指出的那样,我注意到线程会升温,也就是说,它们会随着重复执行而变得更快。我想了解为什么会发生这种情况,以及它是否与 Java 本身有关,或者它是否是每个多线程程序的常见行为。

示例代码(由 Peter Lawrey 编写)如下:

for (int i = 0; i < 20; i++) {
    ExecutorService es = Executors.newFixedThreadPool(1);
    final double[] d = new double[4 * 1024];
    Arrays.fill(d, 1);
    final double[] d2 = new double[4 * 1024];
    es.submit(new Runnable() {
    @Override
    public void run() {
        // nothing.
    }
    }).get();
    long start = System.nanoTime();
    es.submit(new Runnable() {
    @Override
    public void run() {
        synchronized (d) {
            System.arraycopy(d, 0, d2, 0, d.length);
        }
    }
    });
    es.shutdown();
    es.awaitTermination(10, TimeUnit.SECONDS);
    // get a the values in d2.
    for (double x : d2) ;
    long time = System.nanoTime() - start;
    System.out.printf("Time to pass %,d doubles to another thread and back was %,d ns.%n", d.length, time);
}

结果:

Time to pass 4,096 doubles to another thread and back was 1,098,045 ns.
Time to pass 4,096 doubles to another thread and back was 171,949 ns.
 ... deleted ...
Time to pass 4,096 doubles to another thread and back was 50,566 ns.
Time to pass 4,096 doubles to another thread and back was 49,937 ns.

即它变得更快并稳定在 50 ns 左右。这是为什么?

如果我运行此代码(20 次重复),然后执行其他操作(假设对先前结果进行后处理并为另一轮多线程做准备),然后在相同Runnable的情况ThreadPool下再执行 20 次重复,它将已经预热,在任何情况?

在我的程序中,我Runnable只在一个线程中执行(实际上每个处理核心一个,它是一个 CPU 密集型程序),然后交替执行其他一些串行处理多次。随着程序的进行,它似乎并没有变得更快。也许我可以找到一种方法来加热它……</p>

4

2 回答 2

9

与 JVM 相比,正在升温的不是线程。

JVM 具有所谓的 JIT(即时)编译功能。当程序运行时,它会分析程序中发生的事情并即时优化它。它通过获取 JVM 运行的字节码并将其转换为运行速度更快的本机代码来实现这一点。它可以以最适合您当前情况的方式执行此操作,因为它通过分析实际运行时行为来执行此操作。这可以(不总是)导致很大的优化。甚至比一些在没有这些知识的情况下编译为本机代码的程序更是如此。

您可以在http://en.wikipedia.org/wiki/Just-in-time_compilation阅读更多内容

当代码加载到 CPU 缓存中时,您可以在任何程序上获得类似的效果,但我相信这将是一个较小的差异。

于 2011-03-04T20:00:01.913 回答
1

我看到线程执行最终可以更快的唯一原因是:

  • 内存管理器可以重用已经分配的对象空间(例如,让堆分配填满可用内存,直到达到最大内存 -Xmx属性)

  • 工作集在硬件缓存中可用

  • 重复操作可能会创建操作,编译器可以更轻松地重新排序以优化执行

于 2011-03-04T20:00:41.547 回答