对于这个问题应该用JVM的工作步骤来解释,首先看如下代码
class X {
Y b = new Y();
X() {
System.out.print("X");
}
}
class Y {
Y() {
System.out.print("Y");
}
}
public class Z extends X {
Y y = new Y();
Z() {
System.out.print("Z");
}
public static void main(String[] args) {
new Z(); //结果是 YXYZ
}
}
这段代码的执行过程如下:
1.在栈内存定义变量此时为初始值,定义方法。基本数据类型为0 ,引用数据类型为null
2. 调用父类构造方法,定义父类的属性和方法(如果子类已经重写父类的方法 这时不会被覆盖,整个过程不会发生任何覆盖的情 况)
3. 给父类的变量赋值。(此处即new了一个Y对象)
4. 执行父类构造方法中其他语句(此时它自己变量已经初始化和赋值完成,貌似很合理)
5. 给自己变量赋值(在堆内存创建对象或常量)(此处也是new了一个Y对象)
6. 执行构造方法中其他语句
铺垫的小知识:
第一个:成员变量有基本类型和引用类型的。
class Demo {
//基本类型
int x = 10;
//引用类型
Student s = new Student();
}
第二个:类的初始化过程
加载class文件
堆中开辟空间
变量的默认初始化
变量的显示初始化
构造代码块初始化
构造方法初始化
第三个:遇到extends,就要知道,先初始化父类数据,然后初始化子类数据。
分层初始化。
super在这里仅仅表示要先初始化父类数据。