jvm之方法调用过程

方法调用阶段的任务是确定要调用的方法的版本。class文件的编译与其他程序语言不同,它不包含连接步骤。class文件中存储的只是方法的符号引用。需要在类加载阶段(解析)或者运行时(委派)才能确定方法的直接引用。

一、解析

1.非虚方法

有一些方法在编译完成时就可以确定调用版本了。比如私有方法、静态方法、实例构造器、父类方法、被final修饰的方法,他们不会被子类重写,因此在编译期就可以确定要执行的版本。这类方法的调用被称为解析。

调用不同类型的方法,字节码指令集设计了不同的指令:

  • invokestatic :调用静态方法
  • invokespecial :调用<init>()、私有方法、父类方法
  • invokevirtual :调用虚方法
  • invokeinterface :调用接口方法
  • invokedynamic : 可以由用户程序决定调用哪个方法

静态方法、私有方法、实例构造器、父类方法以及被final修饰的方法(由invokevirtual调用)这五个方法在类加载阶段就可以解析为直接引用,这些方法被称为 非虚方法

2.静态分派

静态分派是指根据变量的静态类型决定调用方法的版本。这一步是在编译期完成的。

public class StaticDispatch {
	static abstract class Human {
	}
	static class Man extends Human {
	}
	static class Woman extends Human {
	}
	public void sayHello(Human guy) {
		System.out.println("hello,guy!");
	}
	public void sayHello(Man guy) {
		System.out.println("hello,gentleman!");
	}
	public void sayHello(Woman guy) {
		System.out.println("hello,lady!");
	}
	public static void main(String[] args) {
		Human man = new Man();
		Human woman = new Woman();
		StaticDispatch sr = new StaticDispatch();
		sr.sayHello(man);
		sr.sayHello(woman);
	}
}

运行结果

hello,guy!
hello,guy!

在这段代码中,Human是静态类型、Man和Woman是实际类型。在编译期间,编译器就根据静态类型把sayHello(Human) 符号引用写在了 invokevirtual的参数中。

二、动态分派

动态分派根据变量的实际类型调用方法。jvm通过invokevirtual实现动态分派。

在这里插入图片描述
这里objectref是实例对象的引用,也就是说invokevirtual要调用实际对象的方法。下面是invokevirtual的参数列表,要求操作数栈存放这些参数:
在这里插入图片描述

我们看下面这段代码:

class FieldHasNoPolymorphic {
    static class Father {
        public int money = 1;
        public Father() {
            money = 2;
            showMeTheMoney();
        }
        public void showMeTheMoney() {
            System.out.println("I am Father, i have $" + money);
        }
    }
    static class Son extends Father {
        public int money = 3;
        public Son() {
            money = 4;
            showMeTheMoney();
            fun();
        }
        public void showMeTheMoney() {
            System.out.println("I am Son, i have $" + money);
        }
        public void fun(){

        }
    }
    public static void main(String[] args) {
        Father gay = new Son();
        System.out.println("This gay has $" + gay.money);
    }
}

输出结果:

I am Son, i have $0
I am Son, i have $4
This gay has $2

执行过程:

  1. Father gay = new Son();

对应字节码指令:

0: new           #2                  // class FieldHasNoPolymorphic$Son
3: dup
4: invokespecial #3                  // Method FieldHasNoPolymorphic$Son."<init>":()V

new指令分配内存并返回引用,然后用invokespecial调用Son.<init>()方法。
这里new返回的引用作为参数(objectref)给invokespecial。
在这里插入图片描述

  1. 执行Son的构造方法

java代码:

        public Son() {
            money = 4;
            showMeTheMoney();
        }

对应字节码指令:

public FieldHasNoPolymorphic$Son();
  descriptor: ()V
  flags: ACC_PUBLIC
  Code:
    stack=2, locals=1, args_size=1
       0: aload_0
       1: invokespecial #1                  // Method FieldHasNoPolymorphic$Father."<init>":()V
       4: aload_0
       5: iconst_3
       6: putfield      #2                  // Field money:I
       9: aload_0
      10: iconst_4
      11: putfield      #2                  // Field money:I
      14: aload_0
      15: invokevirtual #3                  // Method showMeTheMoney:()V
	... ...
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0      23     0  this   LFieldHasNoPolymorphic$Son;

先用invokespecial执行Father.<init>()方法。这里aload_0把局部变量表第0个变量加载到操作数栈并作为参数传递给Father.<init>()。

注意:java代码中Son的构造方法是没有参数的,但是在字节码文件中stack=2, locals=1, args_size=1 说参数数目为1。这个参数对应局部变量表中Name为this的变量,这个this是第1步中new指令创建的。也就是说第1步new返回的引用最终传给了Father.<init>()

  1. 执行Father构造方法

Java代码:

        public Father() {
            money = 2;
            showMeTheMoney();
        }

对应字节码

public FieldHasNoPolymorphic$Father();
    descriptor: ()V
    flags: ACC_PUBLIC
    Code:
      stack=2, locals=1, args_size=1
         0: aload_0
         1: invokespecial #1                  // Method java/lang/Object."<init>":()V
         4: aload_0
         5: iconst_1
         6: putfield      #2                  // Field money:I
         9: aload_0
        10: iconst_2
        11: putfield      #2                  // Field money:I
        14: aload_0
        15: invokevirtual #3                  // Method showMeTheMoney:()V
        18: return
		... ...
	   LocalVariableTable:
	    Start  Length  Slot  Name   Signature
	        0      19     0  this   LFieldHasNoPolymorphic$Father;

标号14、15的字节码指令把this作为参数,用invokevirtual执行showMeTheMoney方法。

根据上面引用的虚拟机规范的描述,此时要调用this对应的类C也就是Son的方法。而 Son中的money变量仅仅在准备阶段赋为默认值0。所以第一行打印输出:

I am Son, i have $0

Father()执行完后,紧接着调用Son(),输出:

I am Son, i have $4

最后,在main方法中,直接输出Father的money变量:

This gay has $2
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值