下面是代码

package org.nagark;


class baseClass{

    public void callMtd(int i){
        System.out.println("integer method"+i);
    }

    public void callMtd(double d){
        System.out.println("double method"+d);
    }
}

public class OverRidingExample {

    /**
     * @param args
     */
    public static void main(String[] args) {
        baseClass bc = new baseClass();
        bc.callMtd(10/3);
    }

}


OverRidingExample类中,我用参数10/3调用baseClass.callMtd方法(如您在代码中看到的)。由于callMtd在baseClass中已重载,因此默认情况下应调用哪个版本?因为它是重载方法绑定应该在编译时发生,但是10/3的计算是否可以在编译时发生?

最佳答案

10/33相同,因此将调用int版本。

仅当不存在int参数化方法时,double才会强制转换为int

10/3的计算将在编译时进行,因为它满足常量表达式的definition(感谢holger)。

该方法及其完整签名(包括参数类型)始终在编译时解析。例如,如果您使用某个jar并从此jar中的任何类调用某个方法methodName(String x),并且此方法将签名(广泛使用)更改为更现代的版本替换为methodName(Object x)时(没有问题,是'是吗?),执行将失败。

顺便说一下,如果不确定,总是可以查看生成的OverRidingExample.class的字节码:

$ javap -c  OverRidingExample.class
Compiled from "OverRidingExample.java"
public class org.nagark.OverRidingExample {
  public org.nagark.OverRidingExample();
    Code:
       0: aload_0
       1: invokespecial #1                  // Method java/lang/Object."<init>":()V
       4: return

  public static void main(java.lang.String[]);
    Code:
       0: new           #2                  // class org/nagark/baseClass
       3: dup
       4: invokespecial #3                  // Method org/nagark/baseClass."<init>":()V
       7: astore_1
       8: aload_1
       9: iconst_3
      10: invokevirtual #4                  // Method org/nagark/baseClass.callMtd:(I)V
      13: return
}


您可以看到,这是:


没有除法,只有常数的加载(我想等于3)
调用int参数化的方法。优质教育


如果您不熟悉字节码,则可以阅读wiki文章。

10-06 09:06