首先引出一个问题,是在其他博客上看到的,在这里也分享给大家。
public class Parent{
public void test()
{}
public Parent(){
test();
}
public static void main(String[] args){
new Child();
}
}
class Child extends Parent{
private int instanceValue = 20;
public void test(){
System.out.println("instance value is: " + instanceValue);
}
}
先看一遍代码,看看打印的结果是多少?
想要得到正确的结果,那么必须要知道在java中对象实例化的过程
举例:Student s = new Student();
1)JVM加载Student.class文件,如果Student有直接父类(通过extends直接继承,Object除外),也一并加载
2)在堆栈中分配地址,在堆中给new出来的Student分配地址,在栈中创建引用变量 s
3)在堆中给new出来的Student的属性进行默认初始化,子类构造函数进栈
4)子类构造函数第一句其实省略了super(); 所以在子类初始化之前,先对父类进行初始化
5)父类对象属性默认初始化,父类无参构造进栈,若父类还有父类,再对父类的父类进行初始化,若没有,父类对象属性显示初始化,父类构造代码块初始化,父类构造函数具体初始化
6)父类初始化完毕,子类对象属性显示初始化,子类构造代码块初始化,再执行子类构造函数具体初始化
7)初始化完毕后,将此对象在堆中的地址值赋给栈中引用变量 s
我们再来看一下刚开始给大家抛出的那个问题
1)jvm从main方法开始执行
2)首先执行new Child()创建Child对象
3)Child对象属性默认初始化 instanceValue = 0,因为int的默认值为0
4)Child无参构造函数进栈,
5)Parent没有属性,也没有构造代码块,所以Parent构造函数直接进栈
6)构造函数调用test方法,Parent自己有test方法,但是Child也有test方法,Parent的test被覆盖,这里调用的是子类的test方法,此时instanceValue = 0,还没有被显示初始化,所以打印出来的是0
7)Parent构造方法执行完毕,Child对象属性显示初始化instanceValue = 20,此时的instanceValue才是20
8)Child方法执行完毕,结束
以上观点为本人查找相关资料总结,欢迎各位大佬指教!