从二进制兼容的角度来看,一个方法由四部分构成,分别是:方法的名称,返回值类型,参数,方法是否为static。改变这四个项目中的任意一个,对jvm而言它已经变成了另一个方法。
以“boolean isvalid()”方法为例,如果让isvalid接收一个date参数,变成“boolean isvalid(date when)”,修改后的类不能直接替换原有的类,试图访问新类的isvalid()方法只能得到类似下面的错误信息:java.lang.nosuchmethoderror: ticket.isvalid()z。jvm用“()z”这个符号表示方法不接受参数且返回一个boolean。关于这一问题,下文将有更详细的说明。
jvm利用一种称为虚拟方法调度(virtual method dispatch)的技术判断要调用的方法体,它根据被调用方法所在的实际实例来决定要使用的方法体,可以看作一种扩展的延迟绑定策略。
如果该类没有提供一个名称、参数、返回值类型完全匹配的方法,它就使用从超类继承的方法。由于java的二进制兼容性规则,这种继承实际上在运行期间确定,而不是在编译期间确定。假设有下面几个类:
class poem {
void perform() {
system.out.println("白日依山尽");
} }
class shakespearepoem extends poem {
void perform() {
system.out.println("to be or not to be.");
} }
class hamlet extends shakespearepoem { }
那么,
poem poem = new hamlet();
poem.perform();
将输出“to be or not to be.”。这是因为perform的方法体是运行时才确定的。虽然hamlet没有提供perform的方法体,但它从shakespearepoem继承了一个。至于为何不用poem定义的perform方法,那是因为shakespearepoem定义的perform已经覆盖了它。我们可以随时修改hamlet,却无需重新编译shakespearepoem,如下例所示:
class hamlet extends shakespearepoem {
system.out.println("连一支耗子都没闹");
}
现在,前面的例子将输出“连一支耗子都没闹”。但是,
poem poem = new shakespearepoem();
poem.perform();
这段代码的输出结果是“to be or not to be.”如果我们删除shakespearepoem的内容,同样的代码将输出“白日依山尽”。
闽公网安备 35060202000074号