方法调用指令
方法调用指令有5种
invokestatic
调用静态方法
invokespecial
调用私有实例方法, 构造器, 以及使用super关键字调用父类的实例方法或者构造器和所实现接口的默认方法.
invokevirtual
调用非私有方法
invokeinterface
调用接口方法
invokedynamic
调用动态方法
invokestatic例子
1 2 3 4 5 6 7 8 9 10 11
| public class Main { public static void main(String[] args) { A.test(); } }
class A { public static String test() { return "A test"; } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| public static void main(java.lang.String[]); descriptor: ([Ljava/lang/String;)V flags: ACC_PUBLIC, ACC_STATIC Code: stack=1, locals=1, args_size=1 0: invokestatic #2 // Method com/yuda/test/A.test:()Ljava/lang/String; 3: pop 4: return LineNumberTable: line 10: 0 line 11: 4 LocalVariableTable: Start Length Slot Name Signature 0 5 0 args [Ljava/lang/String;
|
invokespecial例子
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
| public class Main extends A{ public static void main(String[] args) {
} public void testM (){ super.test(); testM2(); new A(); super.run(); } private void testM2(){
} }
class A implements Runnable{ public String test() { return "A test"; }
@Override public void run() {
} }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
| public void testM(); descriptor: ()V flags: ACC_PUBLIC Code: stack=2, locals=1, args_size=1 0: aload_0 1: invokespecial #2 // Method com/yuda/test/A.test:()Ljava/lang/String; 4: pop 5: aload_0 6: invokespecial #3 // Method testM2:()V 9: new #4 // class com/yuda/test/A 12: dup 13: invokespecial #1 // Method com/yuda/test/A."<init>":()V 16: pop 17: aload_0 18: invokespecial #5 // Method com/yuda/test/A.run:()V 21: return LineNumberTable: line 13: 0 line 14: 5 line 15: 9 line 16: 17 line 17: 21 LocalVariableTable: Start Length Slot Name Signature 0 22 0 this Lcom/yuda/test/Main;
|
invokevirtual例子
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| public class Main{ public static void main(String[] args) {
}
public void testM(A a) { a.test(); } }
class A { public String test() { return "A test"; } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
| public void testM(com.yuda.test.A); descriptor: (Lcom/yuda/test/A;)V flags: ACC_PUBLIC Code: stack=1, locals=2, args_size=2 0: aload_1 1: invokevirtual #2 // Method com/yuda/test/A.test:()Ljava/lang/String; 4: pop 5: return LineNumberTable: line 14: 0 line 15: 5 LocalVariableTable: Start Length Slot Name Signature 0 6 0 this Lcom/yuda/test/Main; 0 6 1 a Lcom/yuda/test/A;
|
invokeinterface例子
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
| public class Main { public static void main(String[] args) {
}
public void testM(A a) { a.run(); } }
class A implements Runnable { @Override public void run() {
} }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| public void testM(com.yuda.test.A); descriptor: (Lcom/yuda/test/A;)V flags: ACC_PUBLIC Code: stack=1, locals=2, args_size=2 0: aload_1 1: invokevirtual #2 // Method com/yuda/test/A.run:()V 4: return LineNumberTable: line 14: 0 line 15: 4 LocalVariableTable: Start Length Slot Name Signature 0 5 0 this Lcom/yuda/test/Main; 0 5 1 a Lcom/yuda/test/A;
|
invokedynamic例子
1 2 3 4 5 6 7 8 9
| public class Main { public static void main(String[] args) { Function<String, String> stringStringFunction = Main::apply; }
private static String apply(String i) { return i; } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| public static void main(java.lang.String[]); descriptor: ([Ljava/lang/String;)V flags: ACC_PUBLIC, ACC_STATIC Code: stack=1, locals=2, args_size=1 0: invokedynamic #2, 0 // InvokeDynamic #0:apply:()Ljava/util/function/Function; 5: astore_1 6: return LineNumberTable: line 13: 0 line 14: 6 LocalVariableTable: Start Length Slot Name Signature 0 7 0 args [Ljava/lang/String; 6 1 1 stringStringFunction Ljava/util/function/Function; LocalVariableTypeTable: Start Length Slot Name Signature 6 1 1 stringStringFunction Ljava/util/function/Function<Ljava/lang/String;Ljava/lang/String;>;
|
静态绑定与动态绑定
- 静态绑定 : 在编译时时能确定目标方法
invokestatic
, invokespecial
- 动态绑定 : 需要在运行时根据调用者的类型动态识别
invokevirtual
, invokeinterface
, invokedynamic
invokevirtual,invokeinterface
invokevirtual,invokeinterface 功能上类似, 都是执行一个方法, 但是invokevirtual
用来调用对象的方法,invokeinterface
用来调用接口的方法. JVM是从效率上考虑这件事的, 要想方法执行快, 就要快速找到这个方法, 每个类文件都关联着一个虚方法表(virtual method table), 这个记录了每个方法的位置信息. 例如A,B类有继承关系, B继承了A, B的虚方法表和A的虚方法表对应, 如果B覆盖了A的某个方法, 只用改变B虚方法表相应位置的函数链接引用就可以了, 如果B新加了方法, 只用在B的虚方法表后面增加相应的记录即可.
invokevirtual
做了这样的优化: 它可以通过虚方法表的index
, 来找到需要执行的方法.
invokeinterface
是针对接口的方法, 由于接口相对于继承的区别, 这种优化在接口上无法实现, 例如: B继承了A, B同时实现了X接口, C仅实现了X接口, 那么B和C的虚方法表中对于接口的方法, 就无法通过位置关系对应起来.
- Java 子类会继承父类的 vtable。Java 所有的类都会继承 java.lang.Object 类,Object 类有 5 个虚方法可以被继承和重写。当一个类不包含任何方法时,vtable 的长度也最小为 5,表示 Object 类的 5 个虚方法
- final 和 static 修饰的方法不会被放到 vtable 方法表里
- 当子类重写了父类方法,子类 vtable 原本指向父类的方法指针会被替换为子类的方法指针
- 子类的 vtable 保持了父类的 vtable 的顺序
invokedynamic
其余4个指令, 都把方法的定位规则固化到了虚拟机内部, 而invokedynamic
开放了限制, 如何定位方法交给用户代码去决定. 流程如下:
- 首次执行
invokedynamic
指令时, 会执行一个引导方法.
- 引导方法返回一个
CallSite
对象, 其内部进行方法的定位, 返回一个MethodHandler对象, 可以重复使用MethodHandler对象
CallSite
对象发生变化后, 将会重新执行定位方法的操作, 这些都在运行期进行.
1 2 3 4 5 6 7 8 9
| public class Main { public static void main(String[] args) { Function<String, String> stringStringFunction = Main::apply; }
private static String apply(String i) { return i; } }
|
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59
| ....略 #25 = Utf8 Main.java #26 = NameAndType #5:#6 // "<init>":()V #27 = Utf8 BootstrapMethods #28 = MethodHandle #6:#35 // invokestatic java/lang/invoke/LambdaMetafactory.metafactory:(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/CallSite; #29 = MethodType #36 // (Ljava/lang/Object;)Ljava/lang/Object; #30 = MethodHandle #6:#37 // invokestatic com/yuda/test/Main.apply:(Ljava/lang/String;)Ljava/lang/String; #31 = MethodType #21 // (Ljava/lang/String;)Ljava/lang/String; #32 = NameAndType #20:#38 // apply:()Ljava/util/function/Function; #33 = Utf8 com/yuda/test/Main #34 = Utf8 java/lang/Object #35 = Methodref #39.#40 // java/lang/invoke/LambdaMetafactory.metafactory:(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/CallSite; #36 = Utf8 (Ljava/lang/Object;)Ljava/lang/Object; #37 = Methodref #3.#41 // com/yuda/test/Main.apply:(Ljava/lang/String;)Ljava/lang/String; #38 = Utf8 ()Ljava/util/function/Function; #39 = Class #42 // java/lang/invoke/LambdaMetafactory #40 = NameAndType #43:#47 // metafactory:(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/CallSite; #41 = NameAndType #20:#21 // apply:(Ljava/lang/String;)Ljava/lang/String; #42 = Utf8 java/lang/invoke/LambdaMetafactory #43 = Utf8 metafactory #44 = Class #49 // java/lang/invoke/MethodHandles$Lookup #45 = Utf8 Lookup #46 = Utf8 InnerClasses #47 = Utf8 (Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/CallSite; #48 = Class #50 // java/lang/invoke/MethodHandles #49 = Utf8 java/lang/invoke/MethodHandles$Lookup #50 = Utf8 java/lang/invoke/MethodHandles
....略
public static void main(java.lang.String[]); descriptor: ([Ljava/lang/String;)V flags: ACC_PUBLIC, ACC_STATIC Code: stack=1, locals=2, args_size=1 0: invokedynamic #2, 0 // InvokeDynamic #0:apply:()Ljava/util/function/Function; 5: astore_1 6: return LineNumberTable: line 13: 0 line 14: 6 LocalVariableTable: Start Length Slot Name Signature 0 7 0 args [Ljava/lang/String; 6 1 1 stringStringFunction Ljava/util/function/Function; LocalVariableTypeTable: Start Length Slot Name Signature 6 1 1 stringStringFunction Ljava/util/function/Function<Ljava/lang/String;Ljava/lang/String;>;
....略
InnerClasses: public static final #45= #44 of #48; //Lookup=class java/lang/invoke/MethodHandles$Lookup of class java/lang/invoke/MethodHandles BootstrapMethods: 0: #28 invokestatic java/lang/invoke/LambdaMetafactory.metafactory:(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/CallSite; Method arguments: #29 (Ljava/lang/Object;)Ljava/lang/Object; #30 invokestatic com/yuda/test/Main.apply:(Ljava/lang/String;)Ljava/lang/String; #31 (Ljava/lang/String;)Ljava/lang/String;
|
lambda表达式
lambda表达式写法很像匿名内部类的语法糖写法, 但是他们是不同的, lambda表达式是使用invokedynamic
指令完成的.
1 2 3 4 5 6 7 8
| public class Main { public static void main(String[] args) { Runnable r = () -> { System.out.println("hello, lambda"); }; r.run(); } }
|
如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42
| ...略 public static void main(java.lang.String[]); descriptor: ([Ljava/lang/String;)V flags: ACC_PUBLIC, ACC_STATIC Code: stack=1, locals=2, args_size=1 0: invokedynamic #2, 0 // InvokeDynamic #0:run:()Ljava/lang/Runnable; 5: astore_1 6: aload_1 7: invokeinterface #3, 1 // InterfaceMethod java/lang/Runnable.run:()V 12: return LineNumberTable: line 10: 0 line 13: 6 line 14: 12 LocalVariableTable: Start Length Slot Name Signature 0 13 0 args [Ljava/lang/String; 6 7 1 r Ljava/lang/Runnable;
private static void lambda$main$0(); descriptor: ()V flags: ACC_PRIVATE, ACC_STATIC, ACC_SYNTHETIC Code: stack=2, locals=0, args_size=0 0: getstatic #4 // Field java/lang/System.out:Ljava/io/PrintStream; 3: ldc #5 // String hello, lambda 5: invokevirtual #6 // Method java/io/PrintStream.println:(Ljava/lang/String;)V 8: return LineNumberTable: line 11: 0 line 12: 8 } SourceFile: "Main.java" InnerClasses: public static final #57= #56 of #60; //Lookup=class java/lang/invoke/MethodHandles$Lookup of class java/lang/invoke/MethodHandles BootstrapMethods: 0: #27 invokestatic java/lang/invoke/LambdaMetafactory.metafactory:(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/CallSite; Method arguments: #28 ()V #29 invokestatic com/yuda/test/Main.lambda$main$0:()V #28 ()V
|
可以看到他和 java/lang/invoke/LambdaMetafactory
中的 metafactory
方法有点关系. 方法如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| public static CallSite metafactory(MethodHandles.Lookup caller, String invokedName, MethodType invokedType, MethodType samMethodType, MethodHandle implMethod, MethodType instantiatedMethodType) throws LambdaConversionException { AbstractValidatingLambdaMetafactory mf; mf = new InnerClassLambdaMetafactory(caller, invokedType, invokedName, samMethodType, implMethod, instantiatedMethodType, false, EMPTY_CLASS_ARRAY, EMPTY_MT_ARRAY); mf.validateMetafactoryArgs(); return mf.buildCallSite(); }
|
caller
:表示具有调用者的可访问权限的查找上下文。当与 invokedynamic 一起使用时,它会被虚拟机自动堆叠。
invokedName
:要实现的方法的名称。当与 invokedynamic 一起使用时,它由 invokedynamic 结构的 NameAndType 提供,并由虚拟机自动堆叠。
invokedType
:预期的 CallSite 的签名。参数类型表示捕获变量的类型;返回类型是要实现的接口。当与 invokedynamic 一起使用时,它由 invokedynamic 结构的 NameAndType 提供,并由虚拟机自动堆叠。如果实现方法是一个实例方法,并且该签名有任何参数,则调用签名中的第一个参数必须与接收方对应。
samMethodType
:由函数对象实现的方法的签名和返回类型
implMethod
:一个直接方法句柄,描述在调用时应该调用的实现方法(参数类型、返回类型的适当调整,以及在调用参数之前包含捕获的参数)。
instantiatedMethodType
:应该在调用时动态执行的签名和返回类型。这可能与samMethodType相同,也可能是它的专门化。

lambdaClassName
就是新生成的一个类com/yuda/test/Main$$Lambda$1
的类名. 类中一个方法包含了#29 invokestatic com/yuda/test/Main.lambda$main$0:()V
调用了 com/yuda/test/Main
中的lambda$main$0()
方法.
- lambda 表达式声明的地方会生成一个 invokedynamic 指令,同时编译器生成一个对应的引导方法(Bootstrap Method).
- 第一次执行 invokedynamic 指令时,会调用对应的引导方法(Bootstrap Method),该引导方法会调用 LambdaMetafactory.metafactory 方法动态生成内部类.
- 引导方法会返回一个动态调用 CallSite 对象,这个 CallSite 会链接最终调用的实现了 Runnable 接口的内部类.
- lambda 表达式中的内容会被编译成静态方法,前面动态生成的内部类会直接调用该静态方法.
- 真正执行 lambda 调用的还是用 invokeinterface 指令. ————- from 掘金小册