非静态成员变量在调用自己的构造器赋值之前的赋值过程:
1.默认初始化---->2.显示初始化/代码块中初始化【这两个是并列关系看实际谁在前面】---->3.构造器中初始化---->4.有了对象之后可用“对象.属性”或 “对象.方法” 的方式对成员变量进行赋值
class Father {
int x = 10; //首先非静态成员变量会默认初始化为0,然后显示初始化为10
public Father(){
this.print();
x = 20; //然后构造器中初始化为20
}
public void print(){
System.out.println("Father.x = " + x);
}
}
class Son extends Father(){
int x = 30;
public Son(){
this.print(); //2.当执行完隐式的super()后就开始调用自己构造器里的代码,所以这里执行自己的构造器时成员变量显示初始化从0变为30,所以调用this.print()方法打印的结果为Son.x = 30
x = 40; //3.接着又从30赋值变为40【但是由于属性不具备多态性】
}
public void print(){
System.out.println("Son.x = " + x);
}
}
public class SonTest(){
public static void main(String[] args){
//先看简单的new Father()
//Father f = new Father();
//System.out.println(f.x); //输出结果为两行:Father.x = 10
// 20
//再用多态的
Father f = new Son(); //当我们new Son()时1.会去调用Son的无参构造方法,这时Son的无参构造方法第一行有个隐式的super()去调用父类也就是Father的无参构造方法,Father的无参构造方法第一行虽然也有隐式的super()但这里Object没操作先不管,我们直接看Father的无参构造方法调用的this.print();由于子类Son也重写了该方法,所以会直接去调用子类的print()方法,由于这里的非静态成员变量x只进行了默认初始化,所以print()方法打印出来的Son.x = 0
System.out.println(f.x); //输出结果为三行:Son.x = 0
// Son.x = 30
// 20 【这里是因为属性不具备多态性,所以会直接去找父类的,父类的x从0变10最后变为的20】
}
}