理解对象实例化顺序

首选看一个题目, 以下代码的输出结果是什么:

public class Super {
    private String baseName = "super";

    public Super() {
        callName();
    }

    public void callName() {
        System.out.println(baseName);
    }

    public static void main(String[] args) {
        Super ins = new Sub();
    }
}


class Sub extends Super {
    private String baseName = "sub";

    public void callName() {
        System.out.println(baseName);
    }
}

不知道各位看官的第一印象是什么, 反正我的第一印象是输出 父类的baseName = "super".

然而实际情况是这样的, 出乎了我的意料:

null

Process finished with exit code 0

然后你就会想为什么会这样, 相信以下的内容大家都看过好多次了(当然还能记住多少,就不一定了..笑..).

对象实例化顺序:

  (1) 父类静态代码块(包括静态初始化块,静态属性,但不包括静态方法)
  (2) 子类静态代码块(包括静态初始化块,静态属性,但不包括静态方法 )
  (3) 父类非静态代码块( 包括非静态初始化块,非静态属性 )
  (4) 父类构造函数
  (5) 子类非静态代码块 ( 包括非静态初始化块,非静态属性 )
 
  (6) 子类构造函数
然后问题来了:
  new Sub();
  a.首先会初始化父类的非静态代码块,也就是 private String baseName = "super"; 然后会执行父类的构造函数也就是 public Super() { callName() }; 但是这时候有个问题, 在子类中有相同的方法 callName(); 因为实例化的是 子类, 那实际上调用的也是子类的 callName();方法, 输出的应该是子类的baseName; 但是那这时候还没有到子类非静态代码块的执行时间, 因此子类的baseName还没有初始化, 这时候他还没有值, 所以打印的结果就是null.
 
posted @ 2018-04-11 11:54  C_Guangjin  阅读(590)  评论(1编辑  收藏  举报