【Simple Java】字符串是通过“引用”传递的

这是Java中一个很经典的问题,在stack-overflow上有很多类似的问题,然而很多回答都是错的或者回答不完整;如果你不深入思考的话,会认为这个问题很简单,但是当你深入下去,会发现这个问题很容易让人产生困惑。

一段有趣且让人困惑的代码

package simplejava;

public class Q14 {

    public static void change(String x) {
        x = "cd";
    }

    public static void main(String[] args) {
        String x = new String("ab");
        change(x);
        System.out.println(x);
    }

}

结果打印:

ab

C++版本如下:

void change(string &x) {
    x = "cd";
}
int main(){
    string x = "ab";
    change(x);
    cout << x << endl;
}

打印结果:

cd

常见有误的理解

x变量存储了堆中“ab”对象的引用,当x作为一个参数传入到change()方法内部时,仍然指向堆中的“ab”对象,如下所示:

因为Java是按值传递的,x的值是“ab”对象的引用,当方法change()被调用时,创建了一个新的对象“cd”,然后x指向“cd”对象,如下图所示:

这看起来像是一个完美的解释,他们很清楚Java总是按值传递的。但是,问题到底出在哪里呢?

这段代码到底究竟在做什么

上面的解释有若干处错误,为了更加容易的理解该问题,我们还是先理清下整个过程。

当字符串对象“ab”被创建的时候,Java分配了对应大小的内存空间,然后对象被赋值给变量x,事实上是x变量存储的是对象的引用,这个引用是“ab”对象在内存中的地址;

x变量包含了对象的引用,x并不是“ab”对象,而是一个存储了“ab”对象引用(内存地址)的变量。

Java是按值传递的,当x被传入change()方法的时候,事实上传入的是一个x变量的拷贝。然后在方法change()内部创建了另一个对象“cd”,它有一个不同的引用。真正改变的是这个x变量的拷贝,其值变成了“cd”对象的引用,而不是原始的x变量被改变;

注:感觉说得有点混乱,我的理解,先说这个参数x,其相当于一个局部变量,当使用该参数的时候,将会分配一个新的存储位置,将实参拷贝到该位置,并将该拷贝值传递给该方法;

在change()方法内部,执行x = "cd"的时候,这里的x实际上是main方法的x变量的一个拷贝,一开始其存放的是“ab对象”的引用,执行完这段代码后,其值变成“cd”对象的引用,而main方法的x变量并没有改变,存放的仍然是“ab”对象的引用。

另一错误解释

这个问题的原因跟字符串的不变形没任何关系,即使将String替换成StringBuilder对象,结果仍然不变,关键点是变量存储的是对象的引用,而不是对象本身;

解决这个问题的方法

如果真的想改变这个对象的值,

首先这个对象是要可改变的,例如StringBuilder。

其次,我们要保证没有新对象被创建赋值给参数变量,因为Java只能按值传递。

如下代码:

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");
}

译文链接:http://www.programcreek.com/2013/09/string-is-passed-by-reference-in-java/

时间: 2024-11-14 12:32:43

【Simple Java】字符串是通过“引用”传递的的相关文章

Java中String是不是引用传递?

本文转自:http://www.jcodecraeer.com/a/chengxusheji/java/2012/0805/340.html 编者语:书上都说string是引用类型,但事实上我所看到的string和所谓的值类型没有什么区别,但通过看以下的文章,明白了: 1.string a="abc";之后,如果a="xy",则是a并没有改变内存中已经存在的"abc",而是又创建了另外一个实例.实际上相当于:string a=new String

Java中参数的引用传递和值传递

1.一些定义 值传递:把实际传入参数的值,在内存中赋值一份 引用传递:传入的是实际参数的地址 2.传递基本类型的参数 因为Java是值传递,那就不难理解下面的代码不交换任何值 swap(Type arg1, Type arg2) { Type temp = arg1; arg1 = arg2; arg2 = temp; } 3.传递引用变量的参数 Java都是通过引用来操作对象的,所有指向对象的变量都是引用,然而,Java不会把引用传递给方法的参数,而是引用的值: 那么,为什么对象的成员值可以改

Java中String的 &quot;引用&quot; 传递

1.来看一段有趣但又让人困惑的代码片段 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" 2.这段代码真正做了什么呢?来解释一下这个过程 首先,当字符串"ab&q

Java基础_0310:引用传递

引用传递 引用传递是Java之中最让初学者费解的概念,而在实际的开发之中,引用传递又有着非常重要的作用: 引用传递的核心在于同一块堆内存空间被不同的栈内存所指向: 范例:第一道引用传递范例 class Message { private int num = 10; // 定义int基本类型的属性 /** * 本类没有提供无参构造方法,而是提供有参构造,可以接收num属性的内容 * @param num 接收num属性的内容 */ public Message(int num) { this.nu

Java学习---- 数组的引用传递

1. public class ArrayRefDemo01{ public static void main(String args[]){ int temp[] = {1,3,5} ; // 利用静态初始化方式定义数组 fun(temp) ; // 传递数组 for(int i=0;i<temp.length;i++){ System.out.print(temp[i] + ".") ; } } public static void fun(int x[]){ // 接收整型

java对象Integer不能引用传递

/** * The value of the <code>Integer</code>. * * @serial */ private final int value; /** * Constructs a newly allocated <code>Integer</code> object that * represents the specified <code>int</code> value. * * @param valu

java的引用传递

最近看着李兴华讲师的java视频教程学习java,关于java引用传递方面的知识的总结. 基础知识 java的常用内存空间 栈内存空间:保存所有的对象名称(更准确地说是保存了引用的堆内存空间的地址) 堆内存空间:保存具体对象的具体属性内容. 全局数据区:保存static类型的属性 全局代码区:保存所有的方法定义 实例分析 class Person { private String name; private int age; private static String city = "北京&qu

Java Object 引用传递和值传递

Java Object 引用传递和值传递 @author ixenos Java中的引用传递: 除了在将参数传递给方法(或函数)的时候是"值传递",传递对象引用的副本,在任何用"="向引用对象变量赋值的时候都是"引用传递",传递对象的引用给另一个变量. 参数传递,传递引用的副本,这看起来是引用传递,实则是传递了副本,这已经是值传递的概念了: 变量赋值,传递引用,这算引用传递 Java参数传递中没有引用传递都是值传递 1.在 Java 应用程序中永

老生常谈--Java值传递和引用传递

起因 前两天面试被问到了这个问题,虽然之前老早就了解过这个问题,但是并没有深入了解,所以面试的时候一下子慌了,菜是原罪,今天菜鸡来补补基础知识. 其实这个问题一直是被讨论的,常见的三种说法就是,1,Java 值传递引用传递都有,2,只有值传递,3只有引用传递,今天查了很多资料,我发现这个问题并不是随随便便就能说清楚. 先说传参 方法的参数可以分为实参和形参,实参是指被调用时传入的实际的值,在方法调用前就已经初始化完毕.而形参是指方法中用来“承接”实参的参数,它是在这个方法里有效,即作用域.方法执