String在我们编程的过程中用的是非常多的,所以不管是笔试还是面试都是经常会被考到或被问到。下面这段代码,可是看看输出的是什么:
public class StringTutorial {
public void changeStr(String s) {
s = "abc";
}
public static void main(String[] args) {
StringTutorial tutorial = new StringTutorial();
String str = "123";
tutorial.changeStr(str);
System.out.println(str);
}
}
我想大部分人会回答是123
,这也是正确答案,但是肯定会有人回答错误的。下面我们就分析下这个问题。
首先我们知道Java里面都是值传递,其次我们知道对象实例是存储在堆中的,java虚拟机栈则存储的是变量的地址,地址指向堆中实例。
方法区:各个线程共享的内存区域,存储已被虚拟机加载的类信息、常量、静态变量、即时编译的器编译后的代码等数据。运行时常量池是方法区的一部分,用于存放编译期生成的各种字面量和符号引用。
结合上面的说明,我画出了对应的堆栈分布图:
当执行下面的代码时,会在常量池生成123
,并在栈上会有对应的对象引用,地址指向常量池。
String str = "123"`
当调用changeStr
方法时,但是还未执行s = "abc"
代码时,因为java是值传递,所以调用changeStr
方法时,传递的是str
的值,即对应123
的地址。对应堆栈图如下所示:
image.png
当执行s = "abc"
时,常量池会生成对应的“abc”,这个时候s的地址发生了改变,如下图所示:
image.png
从上图我们可以得知,str
的值是没有变化的,还是123
。