题目:指出下面程序运行的结果是
package demo;
public class Example {
String str = new String("good");
char[] ch = {'a','b','c'};
//mian方法对应的栈帧
public static void main(String[] args) {
Example ex = new Example();
ex.change(ex.str,ex.ch);
System.out.println(ex.str + " and");
System.out.println(ex.ch);
}
private void change(String str, char[] ch) {
str = "test ok";
ch[0] = 'g';
}
}
结果:good and gbc
这道题从内存模型来进行分析:
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