java字符串传递_简易Java(10):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字符串。如下图所示:

SimpleJava_0901_string_reference.png

因为Java是“值传递”(pass-by-value),x的值是引用指向的字符串ab。当调用change()方法时,它将创建一个新的字符串对象cd,此时x指向字符串cd。如下图所示:

SimpleJava_0902_chang_string.png

这似乎是一个理由充足的解释。它们明确地表明,Java总是按值传递。但是,错在哪里啊?

3、这段代码究竟干了什么?

上面的解释有一些错误。为了更好地理解这个问题,我们来认真地查看一下整个流程。

当字符串ab被创建时,Java分配了足以容纳字符串对象的内存。接着,这个对象被赋值给了变量x,这个对象也确实被赋予了指向该对象的引用。这个引用就是存储这个对象的内存地址。

变量x包含了一个执行字符串对象的引用。但是,x不是引用本身!它仅仅是一个存储引用(内存地址)的变量。

Java是按值传递。当x被传递给change()方法时,x的值(引用)的副本被传递过去了。change()方法创建了另外一个对象cd,那么它有了一个不同的引用。变量x改变它的引用,指向cd,而不是引用本身。

SimpleJava_0903_string_reference.png

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》走起!

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值