时代总是在不断进步发展着的,很多的东西也都在不断的变化更新着。现在很多编程语言都是大家在不断学习的方向,尤其是java更是热门。很多人对于java的一些入门知识很感兴趣,一起来了解一下吧。
首先我们需要知道的是,java的方法调用有两类,动态方法调用与静态方法调用。
1.静态方法调用是指对于类的静态方法的调用方式,是静态绑定的
2.动态方法调用需要有方法调用所作用的对象,是动态绑定的。
类调用(invokestatic)是在编译时就已经确定好具体调用方法的情况。实例调用(invokevirtual)则是在调用的时候才确定具体的调用方法,这就是动态绑定,也是多态要解决的核心问题。
JVM的方法调用指令有四个,分别是invokestatic,invokespecial,invokesvirtual 和 invokeinterface。前两个是静态绑定,后两个是动态绑定的。
方法表与方法调用,如有类定义Person, Girl, Boy,代码如下所示:
class Person { public String toString() { return "I'm a person."; } public void eat() {} public void speak() {} } class Boy extends Person { public String toString() { return "I'm a boy"; } public void speak() {} public void fight() {} } class Girl extends Person { public String toString() { return "I'm a girl"; } public void speak() {} public void sing() {} }
当这三个类被载入到Java虚拟机之后,方法区中就包含了各自的类的信息。Girl和Boy在方法区中的方法表可表示如下图:
从上图中可以看出来,Girl和Boy的方法表包含继承自Object的方法,继承自直接父类Person的方法及各自新定义的方法。
注意方法表条目指向的具体的方法地址,如Girl继承自Object的方法中,只有toString()指向自己的实现(Girl的方法代码),其余皆指向 Object的方法代码;其继承自于Person的方法eat()和speak()分别指向Person的方法实现和本身的实现。
如果子类改写了父类的方法,那么子类和父类的那些同名的方法共享一个方法表项。因此,方法表的偏移量总是固定的。所有继承父类的子类的方法表中,其父类所定义的方法的偏移量也总是一个定值。
Person或Object中的任意一个方法,在它们的方法表和其子类Girl和Boy的方法表中的位置(index)是一样的。这样JVM在调用实例方法其实只需要指定调用方法表中的第几个方法即可。
如调用如下代码所示:
class Party { void happyHour() { Person girl = new Girl(); girl.speak(); } }
当编译Party类的时候,生成girl.speak()的方法调用假设为:Invokevirtual#12
设该调用代码对应着girl.speak();#12是Party类的常量池的索引。JVM执行该调用指令的过程如下图所示:
1.在常量池(这里有个错误,上图为ClassReference常量池而非Party的常量池)中找到方法调用的符号引用 。
2.查看Person的方法表,得到speak方法在该方法表的偏移量(假设为15),这样就得到该方法的直接引用。
3.根据this指针得到具体的对象(即girl所指向的位于堆中的对象)。
4.根据对象得到该对象对应的方法表,根据偏移量15查看有无重写(override)该方法,如果重写,则可以直接调用(Girl的方法表的speak项指向自身的方法而非父类);如果没有重写,则需要拿到按照继承关系从下往上的基类(这里是Person类)的方法表,同样按照这个偏移量15查看有无该方法。
以上就是关于java的方法调用方式有哪些以及主要有哪几种的主要内容了。具体的内容还是很详细的。如果你对java知识感兴趣,想要了解更多java常见问题,敬请关注奇Q工具网。
推荐阅读: