
理解Java程序的执行
博客园 2023-04-22 15:41:41
public class Solution { public static void main(String[] args) { Person person = new Person(); person.hello(); }}class Person { public void hello() { System.out.println("hello"); }}
源文件名是 Solution.java,这是因为文件名必须与 public 类的名字相匹配。在一个源文件中,只能有一个公有类,但可以有任意数目的非公有类。
在这个示例程序中包含两个类:Person 类和带有 public 访问修饰符的 Solution 类。Solution 类包含了 main 方法。
(资料图片)
当编译这段源代码的时候,编译器将在目录下创建两个 class 文件:Solution.class 和 Person.class。
将程序中包含 main 方法的类名提供给字节码解释器,以便启动这个程序:java Solution。字节码解释器开始运行 Solution 类的 main 方法中的代码。
理解方法调用下面假设要调用 x.f(args)。下面是调用过程的详细描述:
编译时:编译器査看对象变量的声明类型和方法名,然后获得所有可能被调用的候选方法。编译器査看调用方法时提供的参数类型,然后获得需要调用的方法名字和参数类型。运行时:如果方法是 private 方法、static 方法、final 方法或者构造器方法,那么编译器将可以准确地知道应该调用哪个方法,我们将这种调用方式称为静态绑定(static binding)。与此对应的是,调用的方法依赖于对象变量 x 的实际类型,并且在运行时实现动态绑定。虚拟机预先为每个类创建了一个方法表(method table),方法表中列出了所有方法的签名和实际调用的方法。这样一来,在真正调用方法的时候,虚拟机仅查找方法表就行了。1、编译器査看对象变量的声明类型和方法名,然后获得所有可能被调用的候选方法。假设调用 x.f(param),且对象变量 x 被声明为 C 类型。需要注意的是:有可能存在多个名字为 f,但参数类型不一样的方法。例如,可能存在方法 f(int) 和方法 f(String)。编译器将会一一列举所有 C 类中名为 f 的方法和其父类中访问属性为 public 且名为 f 的方法(父类的私有方法不可访问)。至此,编译器已获得所有可能被调用的候选方法。
2、接下来,编译器将査看调用方法时提供的参数类型,然后获得需要调用的方法名字和参数类型。如果在所有名为 f 的方法中存在一个与提供的参数类型完全匹配,就选择这个方法。这个过程被称为重载解析(overloading resolution)。例如,对于调用 x.f("Hello") 来说,编译器将会挑选 f(String),而不是 f(int)。由于允许类型转换(int 可以转换成 double,Manager 可以转换成 Employee 等),所以这个过程可能很复杂。如果编译器没有找到与参数类型匹配的方法,或者发现经过类型转换后有多个方法与之匹配,就会报告一个错误。至此,编译器已获得需要调用的方法名字和参数类型。
如果方法是 private 方法、static 方法、final 方法或者构造器方法,那么编译器将可以准确地知道应该调用哪个方法,我们将这种调用方式称为静态绑定(static binding)。与此对应的是,调用的方法依赖于对象变量 x 的实际类型,并且在运行时实现动态绑定。在我们列举的示例中,编译器采用动态绑定的方式生成一条调用 f(String) 的指令。
当程序运行,并且采用动态绑定调用方法时,虚拟机一定调用与对象变量 x 所引用的对象的实际类型最合适的那个类的方法。假设 x 的实际类型是 D,D 是 C 类的子类。如果 D 类定义了 f(String) 方法,虚拟机就直接调用 D 类的 f(String) 方法;否则(D 类中没有定义 f(String) 方法),将在 D 类的父类中寻找 f(String),以此类推。
每次调用方法都要进行搜索,时间开销相当大。因此,虚拟机预先为每个类创建了一个方法表(method table),方法表中列出了所有方法的签名(方法名、参数类型)和实际调用的方法。这样一来,在真正调用方法的时候,虚拟机仅查找方法表就行了。
在前面的例子中,虚拟机搜索 D 类的方法表,以便寻找与调用 f(Sting) 相匹配的方法。这个方法既有可能是 D.f(String),也有可能是 C.f(String),这里的 C 是 D 的父类。这里需要提醒一点,如果调用 super.f(param),编译器将对对象变量父类的方法表进行搜索。
方法调用的示例public static void main(String[] args) { Employee e = new Manager("Carl Cracker", 80000, 1987, 12, 15); System.out.println("salary=" + e.getSalary());}
现在,查看一下调用 e.getSalary() 的详细过程。对象变量 e 被声明为 Employee 类型。Employee 类只有一个名叫 getSalary() 的方法,这个方法没有参数。因此,在这里不必担心重载解析的问题。
由于 getSalary() 不是 private 方法、static 方法,也不是 final 方法,所以将采用动态绑定。虚拟机为 Employee 和 Manager 两个类生成方法表。
在 Employee 的方法表中,列出了这个类定义的方法。实际上,下面列出的方法并不完整,Employee 类有一个父类 Object,Employee 类从这个父类中还继承了许多方法,在此我们略去了 Object 的方法。
Employee: getName() -> Employee.getName() getSalary() -> Employee.getSalary() getHireDay() -> Employee.getHireDay() raiseSalary(double) -> Employee.raiseSalary(doubl e)
Manager 的方法表稍微有些不同。其中有三个方法是继承而来的,一个方法是重新定义的(方法的重写),还有一个方法是新增加的。
Manager: getName() -> Employee.getName() getSalary() -> Manager.getSalary() getHireDay() -> Employee.getHireDay() raiseSalary(double) -> Employee.raiseSalary(doubl e) setBonus(double) -> Manager.setBonus(double)
在运行时,调用 e.getSalary() 的解析过程为:
首先,虚拟机提取对象变量 e 的实际类型的方法表。既可能是 Employee、Manager 的方法表,也可能是 Employee 类的其他子类的方法表。接下来,虚拟机搜索对象变量 e 的实际类型的方法表。此时,虚拟机已经知道应该调用哪个方法。最后,虚拟机调用方法。动态绑定有一个非常重要的特性:无需对现存的代码进行修改,就可以对程序进行扩展。假设增加一个新类 Executive,并且对象变量 e 有可能引用这个类的对象,我们不需要对包含调用 e.getSalary() 的代码进行重新编译。如果 e 恰好引用一个 Executive 类的对象,就会自动地调用 Executive.getSalary() 方法。
参考资料《Java核心技术卷一:基础知识》(第10版)第 5 章:继承 5.1.6 理解方法调用
理解Java程序的执行
今天小编肥嘟来为大家解答以上的问题。心理测试大全以及答案,心理测试大全相信很多小伙伴还不知道,现在让我们一起来看看吧!1、在线提供爱情
据河南省纪委监委4月21日晚消息,平顶山市副市长、新华区委书记安保亮涉嫌严重违纪违法,目前正接受河南省纪委监委纪律审查和监察调查。公开信
大小新闻4月21日讯(YMG全媒体记者杨春娜通讯员程磊摄影报道)4月20日,烟台舰访问烟台,当晚烟台市城管局数字
小伙伴们,你们好,今天小夏来聊聊一篇关于高速脉冲电路,关于高速脉冲电路简述的文章,网友们对这件事情都比较关注,那么现在就
春天做点花生酥送亲友,用面粉就行,花生味浓不发腻,真好吃。花生酥,是家人比较喜欢的小点心,今天我们就来做花生酥,这款花生酥做法很简单
五一假期马上就要来了大家不可避免会遇到单人出行的情况如果在火车的过夜车厢中异性较多,你会介意吗?近日,有女子发帖称自己乘
1、做辅助列咯。2、将这一列的数据复制到后面新的一列,然后使用数据-分列功能,讲A-1-1拆分成三列数据A11,然
衡量一个人的价值,可以参考年龄、社会身份等很多因素,但却不能用这些因素作为借口来贬低人。比如“女人越老越不值钱”这种说法就非常肤浅。
1、激励理论包括那些内容基本概念1。2、马斯洛的需求层次理论2 赫兹伯格的双因素理论3 韦隆的期望理
导读:最近很多玩家都在关注九境传说这款手游,想知道具体的公测时间,九境传说会经过封测、删档内测、不删档测试到最终的公测等几个测试阶段
发明三点式安全带的沃尔沃又搞事情了,这次依然把安全放在了重要位置。刚刚发布的沃尔沃EX90奠定了之后该品牌的发展路线,并且在发布会上打出
鹿鞭膏的配比算是一个很古老的方,很多鹿场都是老一辈人传承下来的,如果没用,早就被拍在沙滩上了,能流传至今,还是能够说明问题的。但是怕
军工股业绩亮眼,主力大幅加仓,行业或迎新一轮景气周期,加仓,国改,军工股,高景气,景气度,军工板块
1、来历:中秋节吃月饼的习俗始于唐朝。2、北宋之时在宫廷内流行,后流传到民间,当时俗称“小饼”和“月团”。3、发展至明朝