服务热线:13616026886

技术文档 欢迎使用技术文档,我们为你提供从新手到专业开发者的所有资源,你也可以通过它日益精进

位置:首页 > 技术文档 > JAVA > 新手入门 > 基础入门 > 查看文档

java二进制兼容规则:方法


  从二进制兼容的角度来看,一个方法由四部分构成,分别是:方法的名称,返回值类型,参数,方法是否为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的内容,同样的代码将输出“白日依山尽”。

扫描关注微信公众号