画图如下代码内存变化图并说明执行过程
class Person{
String name;
int age;
public void show(){
System.out.println("姓名:"+name+",年龄:"+age);
}
}
public class TestPerson {
public static void main(String[] args) {
//创建p1对象
Person p1 = new Person();
p1.age = 24;
p1.name = "张三";
p1.show();
//创建p2对象
Person p2 = new Person();
p2.age = 35;
p2.name = "李四";
p2.show();
}
}
图1为p1执行过程
图2为p2执行过程
内存分为栈,堆,方法区。
第一段p1对象代码
方法区加载代码中的Person类(show方法信息,字符串(张三,李四)),栈中首先为main方法创建栈帧(args:null,p1:null),堆中创建对象假设地址为0x11(name:null,age:0,show:)此时所有参数都是初始化参数,p1相当于地址0x11(也就是对象Person)。【Person p1 = new Person();结束】
下一句代码p1.age = 24; 此时p1=0x11,找到地址0x11,Person对象中age赋值变为24(age:null,变更为age:24),p1.name = "张三"; 再通过p1=0x11找到地址0x11,从Person对象中找到name,name为字符串类型,已事先在方法区中加载好因此name可直接从方法区中获取, p1.show(); 此时栈中再为show方法创建栈帧,show方法已事先在方法区中加载,直接吧方法区中的地址调用即可,调用完之后show[]栈帧删除(因为已经执行结束)
第二段p2对象代码
方法区加载代码中的Person类(show方法信息,字符串(张三,李四)),栈中为main[]方法创建栈帧(args:null,p1:null),堆中创建对象假设地址为0x12(name:null,age:0,show:)参数为初始化参数。
Person p2 = new Person(); p2相当于地址0x12 p2.age = 35; 此时p2等于地址0x12 通过地址0x12找到Person对象中age赋值为35 p2.name = "李四"; 再通过地址0x12找到对象中的name 由于已实现在方法区中加载字符串直接从方法区中将字符串"李四"地址给对象中name即可 p2.show();首先栈中为show[]方法 创建栈帧,再从方法区中获取show[]方法的地址给p2(0x12)中的show;