public class c1 {
public static void main(String[]args){
c1 c=new c2();
//下面这里打印的是c1的成员变量s=AAAA
System.out.println(c.s);
}
public String s="AAAA";
//父类构造函数
public c1(){
call();
}
public void call(){
System.out.println(s);
}
}
class c2 extends c1{
public String s="BBBB";
//子类构造函数
public c2(){
System.out.println(s);
}
//重写父类中的call函数
public void call(){
System.out.println(s);
}
}
输出:
null
BBBB
AAAA
分析:
1.初始化c2()时,先初始化c2的父类:c1()
初始化c1()的静态变量和静态代码块
初始化c2()的静态变量和静态代码块
2.初始化c1()的非静态变量,初始化父类的构造函数。
如果父类的构造函数中调用的方法被子类重写,那么调用的会是子类的方法。
3.初始化c2()的非静态变量,初始化子类的构造函数。
由于父类第一次调用子类的call方法时,子类还没有初始化非静态变量以及构造方法,因此输出是null。
通过父类的引用类型变量指向子类类型对象,访问成员变量时是访问的父类的成员变量。
(另外做个记录:类加载时会为静态变量赋初值,执行静态代码块,但不会执行类的静态方法,类的方法在调用时才会执行)
参考:
https://blog.csdn.net/t_alon/article/details/51752431
https://www.cnblogs.com/hafiz/p/5487814.html