首选看一个题目, 以下代码的输出结果是什么:
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.