Java多态时类的加载顺序

问题

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();
    }
}

输出是什么:

A. null
B. base
C. sub

1.首先,需要明白类的加载顺序。
(1) 父类静态代码块(包括静态初始化块,静态属性,但不包括静态方法)
(2) 子类静态代码块(包括静态初始化块,静态属性,但不包括静态方法 )
(3) 父类非静态代码块( 包括非静态初始化块,非静态属性 )
(4) 父类构造函数
(5) 子类非静态代码块 ( 包括非静态初始化块,非静态属性 )
(6) 子类构造函数
其中:类中静态块按照声明顺序执行,并且(1)和(2)不需要调用new类实例的时候就执行了(意思就是在类加载到方法区的时候执行的)

2.其次,需要理解子类覆盖父类方法的问题,也就是方法重写实现多态问题。
Base b = new Sub();它为多态的一种表现形式,声明是Base,实现是Sub类, 理解为 b 编译时表现为Base类特性,运行时表现为Sub类特性。
当子类覆盖了父类的方法后,意思是父类的方法已经被重写,题中 父类初始化调用的方法为子类实现的方法,子类实现的方法中调用的baseName为子类中的私有属性。

Base b = new Sub();
  1. 类的加载
    1.1 父类的静态代码块:无;
    1.2 子类的静态代码块:无;

  2. 类的初始化
    2.1 初始化父类对象:
    2.1.1 父类的非静态代码块:private String baseName = “base”;
    2.1.2 父类的构造函数:调用 callName 方法;由于 callName 方法被子类重写,所以调用的是子类的 callName 方法;而此时子类的非静态代码块还没有开始执行,也就是 private String baseName = “sub” 没有执行,所以这一步打印出来的字符串是 null;
    2.2 初始化子类对象:
    2.2.1 子类的非静态代码块:private String baseName = “sub”;
    2.2.2 子类的构造函数:默认

你可能感兴趣的:(Java)