转载:https://blog.csdn.net/qq_35038153/article/details/79763157
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)是在类加载的阶段执行的
需要理解子类覆盖父类方法的问题,也就是方法重写实现多态问题。Base b = new Sub(); 它为多态的一种表现形式,声明是Base,实现是Sub类, 理解为 b 编译时表现为Base类特性,运行时表现为Sub类特性。 当子类覆盖了父类的方法后,意思是父类的方法已经被重写,题中 父类初始化调用的方法为子类实现的方法,子类实现的方法中调用的baseName为子类中的私有属性。
因此由1.可知,此时只执行到步骤4.,子类非静态代码块和初始化步骤还没有到,子类中的baseName还没有被初始化。所以此时 baseName为空。 所以为null。
例子2:
class X{ Y y=new Y(); public X(){ System.out.print("X"); } } class Y{ public Y(){ System.out.print("Y"); } } public class Z extends X{ Y y=new Y(); public Z(){ System.out.print("Z"); } public static void main(String[] args) { new Z(); } }输出
YXYZ
初始化过程:
初始化父类中的静态成员变量和静态代码块 ;初始化子类中的静态成员变量和静态代码块 ;初始化父类的普通成员变量和代码块,再执行父类的构造方法;初始化子类的普通成员变量和代码块,再执行子类的构造方法;(1)初始化父类的普通成员变量和代码块,执行 Y y=new Y(); 输出Y (2)再执行父类的构造方法;输出X (3) 初始化子类的普通成员变量和代码块,执行 Y y=new Y(); 输出Y (4)再执行子类的构造方法;输出Z 所以输出YXYZ
