下面代码的输出是什么?
public class Base
{
private String baseName = "base";
public Base()
{
callName();
}
public void callName()
{
System. out. println(baseName);
}
static class Sub extends Base
{
private String baseName = "sub";
public void callName()
{
System. out. println (baseName) ;
}
}
public static void main(String[] args)
{
Base b = new Sub();
}
}
输出的结果是:null
想要理解其中的缘由需要弄懂两个前置知识点
其一
类的加载顺序:
(1) 父类静态代码块(包括静态初始化块,静态属性,但不包括静态方法)
(2) 子类静态代码块(包括静态初始化块,静态属性,但不包括静态方法 )
(3) 父类非静态代码块( 包括非静态初始化块,非静态属性 )
(4) 父类构造函数
(5) 子类非静态代码块 ( 包括非静态初始化块,非静态属性 )
(6) 子类构造函数
其中:类中静态块按照声明顺序执行,并且(1)和(2)不需要调用new类实例的时候就执行了(意思就是在类加载到方法区的时候执行的)
其二
还需要明白子类覆盖父类方法,也就是方法重写实现多态。
Base b = new Sub();它为多态的一种表现形式,声明是Base,实现是Sub类, 理解为 b 编译时表现为Base类特性,运行时表现为Sub类特性。
分析一下程序的运行过程:
执行 Base b = new Sub();语句时,加载父类的构造方法,并执行其中的callName()方法。因为子类覆盖父类的原因,执行的并不是Base中的callName()方法,而是子类重写过的callName()方法。又因为类的加载顺序缘故,此时只执行到了(4),(5)子类非静态代码块还未执行,所以Sub中的baseName为null,即打印结果为null。