包含私有调用的Java内联方法

时间:2018-12-14 21:45:45

标签: java jvm inline private

假设jvm将b.foo()方法标识为hot,它将尝试内联它吗?因为如果这样做,则将在A.main方法中内联调用指令B.bar(),这是被禁止的,因为bar是私有的。在这种情况下,jvm会做什么?

任何具有更多详细信息的文档也将受到赞赏。谢谢

public class A {

    public static void main(String[] args)
    {
        B b = new B();
        for (int i=0; i<99999; i++)
            b.foo();
    }
}

class B {

    public void foo() {
        bar();
    }

    private void bar() { // do something. (is small method)
    }
}

编辑:我之所以问是因为,当我尝试自己(以字节码形式)进行此优化时,我会得到一个有意义的验证错误:

java.lang.VerifyError:错误的特殊调用指令:当前类不能分配给引用类

EDIT2:

public static void main(java.lang.String[]);
descriptor: ([Ljava/lang/String;)V
flags: ACC_PUBLIC, ACC_STATIC
Code:
  stack=2, locals=5, args_size=1
     0: new           #13                 // class B
     3: dup
     4: invokespecial #14                 // Method B."<init>":()V
     7: astore_1
     8: iconst_0
     9: istore_2
    10: iload_2
    11: ldc           #15                 // int 99999
    13: if_icmpge     30
    16: aload_1
    17: astore        4
    19: aload         4
    21: invokespecial #18                 // Method B.bar:()V // VERIFY ERROR HERE
    24: iinc          2, 1
    27: goto          10
    30: return
  StackMapTable: number_of_entries = 2
    frame_type = 253 /* append */
      offset_delta = 10
      locals = [ class B, int ]
    frame_type = 19 /* same */
  LineNumberTable:
    line 5: 0
    line 6: 8
    line 7: 16
    line 14: 19
    line 15: 24
    line 6: 24
    line 8: 30

2 个答案:

答案 0 :(得分:4)

  1. 开头有源代码
  2. java编译器将其转换为字节码
  3. jvm读取字节码,验证其完整性并对其进行解释。即时将“热点”(经常执行的代码块)的字节码编译为运行JVM的CPU的指令集

在Java中,大多数优化将推迟到步骤3中的JIT编译器,以从程序执行期间收集的统计信息中受益。这包括内联。

在验证过程中,解释字节码之前和对其进行JIT编译之前,将检查诸如private之类的访问修饰符。由于已经对访问修饰符进行了检查,因此JIT完全不知道它们的存在,并且内联私有方法不会造成任何困难。

顺便说一句,如果您想检查JIT的功能,则可以使用热点的-XX:+PrintAssembly将机器代码转换回汇编代码以进行检查。

答案 1 :(得分:1)

在Java中,优化通常在JVM级别完成。在运行时,JVM执行一些“复杂”分析以确定要内联的方法。内联可能非常激进,而Hotspot JVM实际上可以内联非最终方法。

java编译器几乎从不内联任何方法调用(JVM在运行时完成所有这些操作)。它们确实内联编译时间常数(例如,最终的静态原始值)。但不是方法。

有关更多资源:

  1. Article: The Java HotSpot Performance Engine: Method Inlining Example

  2. Wiki: Inlining in OpenJDK,虽然未完全填充,但包含指向有用讨论的链接。