8

这是我试图从Java Performance: The Definitive Guide, Page 97关于Escape Analysis主题复制的示例。这可能是应该发生的事情:

  1. getSum()必须变得足够热,并且必须使用适当的 JVM 参数将其内联到调用者main()中。
  2. 由于listsum变量都不会从main()方法中逃脱,它们可以被标记为NoEscape因此 JVM 可以为它们使用堆栈分配而不是堆分配。

但是我通过jitwatch运行它,结果显示它getSum()编译成本地程序集并且没有内联到main(). 更不用说因此堆栈分配也没有发生。

我在这里做错了什么?(我已经把整个代码和热点日志放在这里了。)

这是代码:

import java.math.BigInteger;
import java.util.ArrayList;
import java.util.stream.IntStream;

public class EscapeAnalysisTest {

    private static class Sum {

        private BigInteger sum;
        private int n;

        Sum(int n) {
            this.n = n;
        }

        synchronized final BigInteger getSum() {
            if (sum == null) {
                sum = BigInteger.ZERO;
                for (int i = 0; i < n; i++) {
                    sum = sum.add(BigInteger.valueOf(i));
                }
            }
            return sum;
        }

    }

    public static void main(String[] args) {
        ArrayList<BigInteger> list = new ArrayList<>();
        for (int i = 1; i < 1000; i++) {
            Sum sum = new Sum(i);
            list.add(sum.getSum());
        }
        System.out.println(list.get(list.size() - 1));
    }

}

我使用的JVM参数:

-server
-verbose:gc
-XX:+UnlockDiagnosticVMOptions
-XX:+TraceClassLoading
-XX:MaxInlineSize=60
-XX:+PrintAssembly
-XX:+LogCompilation
4

1 回答 1

5

为了知道为什么有些东西是内联的,你可以在编译日志中查找inline_successandinline_fail标签。

但是,即使要内联某些内容,也必须编译调用者,在您的情况下,您需要在main方法中内联,因此发生这种情况的唯一方法是堆栈替换(OSR)。查看您的日志,您可以看到一些 OSR 编译,但没有一个main方法:您的方法中根本没有足够的工作main

您可以通过增加for循环的迭代次数来解决此问题。通过将其增加到100_000,我得到了第一个 OSR 编译。

对于这样一个小例子,-XX:+PrintCompilation -XX:+PrintInlining而不是整个LogCompilation输出,我看到了:

@ 27   EscapeAnalysisTest$Sum::getSum (51 bytes)   inlining prohibited by policy

这不是很有帮助...但是看一下 HotSpot 源代码就会发现,这可能是因为有一项策略阻止 C1 编译到内联由 C2 编译的 OSR 方法。无论如何,查看 C1 编译完成的内联并不是那么有趣。

添加更多循环迭代(1_000_000在参数上取模Sum以减少运行时间)为我们提供了一个 C2 OSR main

@31   EscapeAnalysisTest$Sum::getSum (51 bytes)   already compiled into a big method  

C2 策略的那一部分是相当自我描述的,并且由InlineSmallCode标志控制:-XX:InlineSmallCode=4k告诉 HotSpot “大方法”的阈值是 4kB 本机代码。在我的机器上足以getSum内联:

  14206   45 %     4       EscapeAnalysisTest::main @ 10 (61 bytes)
                              @ 25   EscapeAnalysisTest$Sum::<init> (10 bytes)   inline (hot)
                                @ 1   java.lang.Object::<init> (1 bytes)   inline (hot)
              s               @ 31   EscapeAnalysisTest$Sum::getSum (51 bytes)   inline (hot)
                                @ 31   java.math.BigInteger::valueOf (62 bytes)   inline (hot)
                                  @ 58   java.math.BigInteger::<init> (77 bytes)   inline (hot)
                                    @ 1   java.lang.Number::<init> (5 bytes)   inline (hot)
                                      @ 1   java.lang.Object::<init> (1 bytes)   inline (hot)
                                @ 34   java.math.BigInteger::add (123 bytes)   inline (hot)
                                  @ 41   java.math.BigInteger::add (215 bytes)   inline (hot)
                                  @ 48   java.math.BigInteger::<init> (38 bytes)   inline (hot)
                                    @ 1   java.lang.Number::<init> (5 bytes)   inline (hot)
                                      @ 1   java.lang.Object::<init> (1 bytes)   inline (hot)
                              @ 34   java.util.ArrayList::add (29 bytes)   inline (hot)
                                @ 7   java.util.ArrayList::ensureCapacityInternal (13 bytes)   inline (hot)
                                  @ 6   java.util.ArrayList::calculateCapacity (16 bytes)   inline (hot)
                                  @ 9   java.util.ArrayList::ensureExplicitCapacity (26 bytes)   inline (hot)
                                    @ 22   java.util.ArrayList::grow (45 bytes)   too big

(请注意,我从来没有使用过MaxInlineSize

作为参考,这里是修改后的循环:

for (int i = 1; i < 1_000_000; i++) {
  Sum sum = new Sum(i % 10_000);
  list.add(sum.getSum());
}
于 2018-03-12T11:18:06.500 回答