内存分析:
首先主线程调用main方法,在主线程栈中main方法执行,main方法对应的栈帧(一个线程一个栈,一个方法一个栈帧)入栈。
1)当代码执行到
Example ex = new Example();
这条语句时,内存模型如下图:
解释:ex作为对象引用保存在栈帧内的局部变量表内,指向在堆中的对象实例,而该对象内的成员str、ch分别指向堆中分配的另外两个对象实例(new String(“good”)、以及数组,数组和对象实例在堆中分配)
2)当主线程代码执行到ex.change(ex.str, ex.ch);这段代码,change方法执行,对应栈帧在主线程栈内入栈,方法内的局部变量存放在栈帧的局部变量表内。
该语句执行时,内存模型如下图:
解释:
Change方法的传入的值
本质上都是字符数组和good的对象实例在堆中的地址,因此change方法内的局部变量ch、str也是分别指向这两个地址,如上图栈帧2中所示。
当change方法内的语句执行完
str = "test ok";
ch[0] = 'g';
这两句时,内存模型如下图:
解释:
1,str = "test ok";
右边这种写法,会直接在运行时常量池内分配字符串“test ok”,并将该地址传递给栈帧2内的局部变量str。
自行百度:
String str = “test”;
和
String str = new String(“test”);
的区别。
2,ch[0] = 'g';
该语句表示将指向的数组对象的第一个字符内容改成’g’。
总结:
由于栈帧1的ex局部变量指向的对象内的成员str指向的对象没有任何变化,因此
System.out.println(ex.str+"and"); 依然打印结果为:good and
而成员ch指向的对象内容有所变化,因此
System.out.println(ex.ch);打印结果为:gbc
纯手打,望点赞。