简易Java(10):Java字符串的“引用传递”
前面有说明“图解Java字符串的不可变性”,后面有解释“为什么Java中的字符串是不可变的? ”。还有一篇与时俱进的“substring()
方法在JDK6和JDK7中的异同 ”。下面,上演第四篇“Java字符串的‘引用传递’”。
这也是一个Java中的一个经典问题。很多类似的问题在StackOverflow被提出。同样有很多不完整甚至不正确的回答。如果不深入思考,这问题很简单。但是,如果我们稍微深究一下,那么她却显得很“迷人”。
1、一段有趣而诡异的代码
我们先看一下下面这段代码:
/** * Coder:D瓜哥,http://www.diguage.com/ */ public class StringDemo { public static void main(String[] args) { String x = new String("ab"); change(x); System.out.println(x); } public static void change(String x) { x = "cd"; } }
输出结果是ab
。
在C++中,有类似代码如下:
/** * Coder:D瓜哥,http://www.diguage.com/ */ void change(string &x) { x = "cd"; } int main() { string x = "ab"; change(x); cout << x << endl; }
那么输出为cd
。
D瓜哥没有开发过C++程序。所以,只是把代码拷贝过来没有补全。大家能看懂就行了,不要深究!
2、常见疑惑问题
x
存储着指向堆栈中字符串ab
的引用。所以,当x
作为参数传递给change()
方法时,它依然指向ab
字符串。如下图所示:
因为Java是“值传递”(pass-by-value),x
的值是引用指向的字符串ab
。当调用change()
方法时,它将创建一个新的字符串对象cd
,此时x
指向字符串cd
。如下图所示:
这似乎是一个理由充足的解释。它们明确地表明,Java总是按值传递。但是,错在哪里啊?
3、这段代码究竟干了什么?
上面的解释有一些错误。为了更好地理解这个问题,我们来认真地查看一下整个流程。
当字符串ab
被创建时,Java分配了足以容纳字符串对象的内存。接着,这个对象被赋值给了变量x
,这个对象也确实被赋予了指向该对象的引用。这个引用就是存储这个对象的内存地址。
变量x
包含了一个执行字符串对象的引用。但是,x
不是引用本身!它仅仅是一个存储引用(内存地址)的变量。
Java是按值传递。当x
被传递给change()
方法时,x
的值(引用)的副本被传递过去了。change()
方法创建了另外一个对象cd
,那么它有了一个不同的引用。变量x
改变它的引用,指向cd
,而不是引用本身。
4、错误解释
第一个代码片段所展示问题和字符串的不变形(“图解Java字符串的不可变性”)没有半毛钱关系。即使将String
替换为StringBuilder
,结果依然一样。问题的核心是变量存储的是引用,而不是引用自身。
5、解决办法
如果你真的想改变对象的值,首先需要确定,对象是可变的,例如,StringBuilder
等。然后,我们需要保证没有新的对象创建并赋值给参数变量。这都是因为Java是按值传递。代码如下
/** * Coder:D瓜哥,http://www.diguage.com/ */ public class StringBuilderDemo { public static void main(String[] args) { StringBuilder x = new StringBuilder("ab"); change(x); System.out.println(x); } public static void change(StringBuilder x) { x.delete(0, 2).append("cd"); } }
《Simple Java》是一本讲解Java面试题的书。讲解也有不少独特之处,为了面试,《简易Java》走起!
原文链接:https://wordpress.diguage.com/archives/97.html
版权声明:非特殊声明均为本站原创作品,转载时请注明作者和原文链接。