对于java单个类的内存分析相信大家都了解,但是如果结合继承,子类与父类是如何存在在内存中的呢,他们的先后调用关系又改是怎样的呢?
首先简单介绍一下对象的初始化过程,比如这个简单的例子
class Student{ int i =20; String name="李四"; static{ System.out.println("我是静态代码块"); } { System.out.println("我是构造代码块"); } void print(){ System.out.println("我是普通方法"); } private String name; private static String country; } public class A6_43{ public static void main(String[] args){ Student s=new Student(); } }
它在内存中的执行顺序是这样的
1.因为new Student()用到了Student类,所以会把它从硬盘上加载进入内存
2.如果有static静态代码块就会随着类的加载而执行,
还有静态成员和普通方法也会随着类的加载而被加载
3.在堆中开辟空间,分配内存地址
4.在堆中建立对象特有属性,并同时对特有属性进行默认初始化
5.对属性进行显示初始化
6.执行构造代码块。对所有对象进行初始化
7.执行对应的构造函数,对对象进行初始化.
8.将内存地址给S(给栈中的变量)
回到正题,如果student继承了person
class Person{ String name="张三"; } class Student extends Person{ String name="李四"; void show(){ System.out.println(name); } } public class A{ public static void main(String[] args){ Student stu=new Student(); stu.show(); } }
那么这个name会返回”李四”。那么张三在哪呢?
因为隐含了this,这里的neme也就是this.name。那么this指向的应该就是堆内存中student的属性,由于student被new所创建,那么他的父类搁哪呢。事实上,他的父类也被创建了,并非是睡眠状态。只是这个时候在堆内存中有两块区域,一个是子类,一个是父类,而this关键字首先访问的是子类,如果子类中没有,再访问父类,都没有就出错。如果想要越过子类直接访问父类就要super关键字,super.属性。
其实,如果子类有属性是跟父类一样的就没有必要再在子类中声明了,讲这些主要是为更好的理解super关键字,也更好的了解java中的内存机制。