为什么对字符串的引用不像其他对象引用?

时间:2011-04-13 17:16:33

标签: java string pass-by-reference


在以下代码中

public class Test {

    public static void main(String[] args){
        int [] arr = new int[]{1,2};
        String b=new String("abc");
        f(b,arr);
        System.out.println(b);
        System.out.println(arr[0]);

    }

    public static void f(String b, int[] arr){

        b+="de";
        b=null;
        arr[0] = 5;
    }
}

为什么字符串的引用变量的行为不像数组的引用变量? 我知道字符串是不可变的,因此对它们的操作会创建新的字符串,但是如何引用字符串以及引用b如何仍然引用旧值,尽管它被更改为引用f()方法中的其他内容。

3 个答案:

答案 0 :(得分:7)

Java中的对象引用按值传递。分配只是更改值,它不会改变原始对象引用。

在您的示例arr[0]已更改,但请尝试arr=null,您将看到该方法返回后无效。

答案 1 :(得分:1)

方法调用在Java中被值调用,关于这一点的争论很长,但我认为我们应该考虑Java的实现语言C / C ++。对象引用只是指向对象的指针,而基元是值。每当调用一个方法时,实际的参数就会被复制到形式参数中。因此,如果更改指针引用另一个对象,原始指针不受此更改的影响,但如果您更改对象本身,主叫方也可以看到更改,因为双方都引用相同的对象..

好吧,在你的例子中,你在被调用的方法中将字符串引用更改为null,但是你正在更改由数组引用引用的对象。这两个操作不一样,因此它们会产生不同的后果...例如,如果您更改下面的代码,它们将在语义上相同的操作..

arr = null;

答案 2 :(得分:0)

您无法更改任何方法的参数,但您可以执行以下操作。

public static void main(String... args) throws IOException {
    String[] strings = {"Hello "};
    addWorld(strings);
    System.out.println("Using an array "+Arrays.toString(strings));

    StringBuilder text = new StringBuilder("Hello ");
    addWorld(text);
    System.out.println("Using a StringBuilder '" + text+"'");
}

private static void addWorld(String[] strings) {
    for(int i=0;i<strings.length;i++)
        strings[i] += "World!";
    strings = null; // doesn't do anything.
}

private static void addWorld(StringBuilder text) {
    text.append("World !!");
    text = null; // doesn't do anything.
}

打印

Using an array [Hello World!]
Using a StringBuilder 'Hello World !!'