class A{
int i = fun();
public A(){
<span style="white-space:pre"> </span>fun();
}
int fun(){
System.out.println(i);
return 1;
}
}
class B extends A{
public B(){
<span style="white-space:pre"> </span>//【此处默认有语句super.();】
fun();
}
int fun(){
System.out.println(i);
return 2;
}
}
public class TestFun {
public static void main(String[] args) {
A a = new B();
}
}
程序的输出结果为:0 2 2
这是OPPO笔试的一道题,当时纠结于构造函数的继承,导致把子类对父类方法的重写给忽略了,最后继承没搞对,重写也没看到,悲剧!以后一定要注意避免这样的情况,踏踏实实的一步步分析。另外,要注重基础的知识,对一个问题理解不清,往往就是基础的原理没有理解清楚。
需要注意三点:整型变量初始化、子类对父类构造函数的继承和子类对父类方法的重写。
整型变量初始化:Java中,如果生成整型变量时没有赋初值,系统会默认初始化,赋值为0。
子类对父类构造函数的继承:构建子类对象时,会自动调用父类的无参构造方法,所以上题中会先调用父类的A(),此时因为父类中有属性int i=fun();,所以会跳转到此处执行,此处的fun()要注意,这个fun()方法到底是父类A中的方法呢,还是子类B中的方法?这就需要用到下面的知识:
子类对父类方法的重写:这里的重写其实可以理解为“覆盖”,即覆盖了子类中从父类继承过来的方法,从此以后此方法对应的就是重写后的方法,而不是父类中的方法。
回到上面的分析,因为生成的是子类对象,所以调用的方法自然就是子类中重写之后的方法,所以程序运行下去就是:打印i的初始值0【此处也有一个知识点,因为i为类属性,所以在调用该类时,JVM会默认赋值】,然后返回2,此时给i赋值i=2,然后再执行fun();语句,即再调用一次子类中的fun(),打印输出2;调用完父类的无参构造方法之后,还要再调用子类的无参构造方法,因为继承了属性i,所以此时子类B中的i=2,打印输出结果为:2。
综上所述,可知程序输出结果为:0 2 2,父类中的fun()方法调用了0次,子类中的fun()方法调用了3次【因为fun()方法被重写】。