转载

黑马程序员————子父类中成员变量的内存分析

对于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中的内存机制。

正文到此结束
Loading...