Java从来没有通过引用,对吧?

可能重复:
Java是“通过引用”吗?

我今天发现了一种不寻常的Java方法:

private void addShortenedName(ArrayList<String> voiceSetList, String vsName) { if (null == vsName) vsName = ""; else vsName = vsName.trim(); String shortenedVoiceSetName = vsName.substring(0, Math.min(8, vsName.length())); //SCR10638 - Prevent export of empty rows. if (shortenedVoiceSetName.length() > 0) { if (!voiceSetList.contains("#" + shortenedVoiceSetName)) voiceSetList.add("#" + shortenedVoiceSetName); } } 

根据我所读到的关于Java传递variables,复杂对象的行为的一切,这段代码应该什么也不做。 所以呃…我在这里错过了什么? 有没有一些微妙的东西丢在我身上,还是这个代码属于dailywtf?

正如Rytmis所说,Java通过价值传递参考。 这意味着你可以合法地调用一个方法的参数的变异方法,但是你不能重新分配它们,并期望值传播。

例:

 private void goodChangeDog(Dog dog) { dog.setColor(Color.BLACK); // works as expected! } private void badChangeDog(Dog dog) { dog = new StBernard(); // compiles, but has no effect outside the method } 

编辑:在这种情况下,这意味着虽然voiceSetList 可能会因为这个方法的结果而改变(它可能会添加一个新的元素),但在方法之外, vsName的改变将不可见。 为了避免混淆,我经常将我的方法参数标记为final ,这使得它们不会被重新分配(意外或不是)在方法内部。 这将保持第二个例子不被编译。

Java通过传递引用 ,所以你得到了引用的副本,但引用的对象是相同的。 因此这个方法修改input列表。

引用本身是通过值传递的。

从Java如何编程,第4版Deitel&Deitel:(第329页)

与其他语言不同,Java不允许程序员select是通过值还是通过引用传递每个参数。 原始数据typesvariables总是按值传递的。 对象不会传递给方法; 相反,对象的引用被传递给方法。 引用本身是通过值传递的 – 引用的副本传递给方法。 当一个方法接收到一个对象的引用时,该方法可以直接操作该对象。

在大学学习Java时使用了这本书。 辉煌的参考。

这是一个很好的文章解释它。 http://www.javaworld.com/javaworld/javaqa/2000-05/03-qa-0526-pass.html

那么,它可以操纵ArrayList – 这是一个对象…如果你正在传递一个对象引用(即使按值传递),对该对象的更改将反映给调用者。 这是个问题吗?

我觉得你很困惑,因为vsName被修改。 但在这种情况下,它只是一个局部variables,与shortenedVoiceSetName完全相同。

我不知道在代码中的确切问题是什么…但是,Java是通过值,但是…数组通过引用传递,因为它们通过没有对象,但只有指针! 数组由指针组成,没有真正的对象,这使得它们非常快速,但也使它们处理危险。 所以你需要克隆它们来获得一个副本,即使这样它只会克隆数组的第一维。

有关更多详细信息,请参阅我的答案: 在Java中,什么是浅拷贝? (也见我的其他答案)

顺便说一句,有一些优点,因为数组只是指针:你可以(ob)使用它们作为同步的对象!