Are arrays passed by value or passed by reference in Java?
Possible Duplicate:
Is Java"pass-by-reference"?
数组不是Java中的原始类型,但它们也不是对象,所以它们是通过值传递还是通过引用传递?它是否依赖于数组包含的内容,例如引用或基元类型?
将数组传递给其他方法时,实际上会复制对该数组的引用。
- 通过该引用对数组内容所做的任何更改都将影响原始数组。
- 但是,将引用更改为指向新数组不会更改原始方法中的现有引用。
看到这个帖子…
Java是"通过引用"还是"按值传递"?
见本工作示例:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 | public static void changeContent(int[] arr) { // If we change the content of arr. arr[0] = 10; // Will change the content of array in main() } public static void changeRef(int[] arr) { // If we change the reference arr = new int[2]; // Will not change the array in main() arr[0] = 15; } public static void main(String[] args) { int [] arr = new int[2]; arr[0] = 4; arr[1] = 5; changeContent(arr); System.out.println(arr[0]); // Will print 10.. changeRef(arr); System.out.println(arr[0]); // Will still print 10.. // Change the reference doesn't reflect change here.. } |
你的问题是基于一个错误的前提。
Arrays are not a primitive type in Java, but they are not objects either ..."
事实上,Java中的所有数组都是Objs1。每一个Java数组类型都有EDOCX1×1作为其超类型,并且继承了EDCOX1、2、API中所有方法的实现。
与所有Java对象一样,数组也是按值传递的…但该值是对数组的引用。
实际的引用传递涉及传递变量的地址,以便可以更新变量。这不是在Java中传递数组时发生的情况。
这里有一些链接可以解释"通过引用"和"通过值"之间的区别:
- http://publib.boulder.ibm.com/infocenter/comphelp/v8v101/topic/com.ibm.xlcpp8a.doc/language/ref/cplr233.htm
- http://www. cf.EdU/~梅尔斯/c++/Notes/参考文献.HTML
相关销售代表问题:
- Java是"通过引用"还是"按值传递"?
历史背景:
短语"pass by reference"最初是"call by reference",用来区分fortran(call by reference)和algol-60(call by value和call by name)的参数传递语义。
在按值调用中,参数表达式被计算为一个值,该值被复制到被调用的方法中。
在引用调用中,参数表达式部分计算为传递给调用方法的"lvalue"(即变量或数组元素的地址)。然后,调用方法可以直接读取和更新变量/元素。
在按名称调用中,实际参数表达式将传递给调用方法(!!)可以多次评估(!!!!)。实现起来很复杂,可以用来(滥用)编写非常难以理解的代码。顾名思义,call-by-name只在algol-60中使用过(谢天谢地!).
更新
实际上,algol-60的名称调用类似于将lambda表达式作为参数传递。问题在于,这些不完全是lambda表达式(它们在实现级别被称为"thunk")可以间接地修改调用过程/函数范围内的变量的状态。这是让他们很难理解的原因之一。(例如,参见Jensen设备上的维基百科页面。)
1。链接的Q&A(Java中的数组以及它们如何存储在内存中)中没有任何状态或暗示数组不是对象。
数组实际上是对象,因此会传递一个引用(引用本身是按值传递的,还不清楚?).快速示例:
1 2 3 4 | // assuming you allocated the list public void addItem(Integer[] list, int item) { list[1] = item; } |
您将看到调用代码对列表的更改。但是,不能更改引用本身,因为它是通过值传递的:
1 2 3 4 |
如果传递一个非空列表,则在该方法返回时,它不会为空。
不,那是错误的。数组是Java中的特殊对象。因此,这就像传递其他对象一样,在其中传递引用的值,而不是引用本身。也就是说,更改被调用例程中数组的引用不会反映在调用例程中。
Java中的所有东西都是通过值传递的。
在数组的情况下,引用被复制到新的引用中,但是请记住Java中的所有东西都是通过值传递的。
请看这篇有趣的文章了解更多信息…
数组的最终讨论在http://docs.oracle.com/javase/specs/jls/se5.0/html/arrays.html 27803上。这清楚地表明Java数组是对象。这些对象的类别在10.8中定义。
语言规范的第8.4.1节,http://docs.oracle.com/javase/specs/jls/se5.0/html/classes.html 40420,描述了如何将参数传递给方法。由于Java语法是从C和C++派生的,所以行为类似。原语类型是按值传递的,与C一样。当传递对象时,对象引用(指针)是按值传递的,这反映了按值传递指针的C语法。参见4.3.1,http://docs.oracle.com/javase/specs/jls/se5.0/html/typesvalues.html 4.3,
实际上,这意味着修改方法中数组的内容会反映在调用范围中的数组对象中,但将新值重新分配给方法中的引用对调用范围中的引用没有影响,这正是C中的结构指针或I中的对象指针所期望的行为。nc++。
术语上的混淆至少有一部分源于高级语言的历史,在C的普遍使用之前。在以前流行的高级语言中,通过地址直接引用内存是应该尽可能避免的事情,并且它被认为是语言提供一个抽象层的工作。这使得语言必须显式支持从子例程(不一定是函数)返回值的机制。这种机制在正式意义上是指"通过引用"。
当引入C时,它附带了一个简化的过程调用概念,其中所有参数都是仅输入的,返回给调用方的唯一值是函数结果。但是,传递引用的目的可以通过显式和广泛使用指针来实现。由于它具有相同的用途,将指针作为对值的引用传递的实践通常被通俗地称为传递引用。如果例程调用的语义要求通过引用传递参数,则C的语法要求程序员显式传递指针。按值传递指针是在C中实现按引用传递语义的设计模式。
由于在C中的原始指针似乎唯一的目的是创建崩溃的bug,随后的开发,尤其是Java,已经试图返回更安全的手段来传递参数。然而,C的主导地位使得开发人员有义务模仿C编码的常见风格。结果是引用被类似地传递给指针,但是被使用更多的保护来实现,以使它们更安全。另一种选择可能是像艾达这样的语言的丰富语法,但是这会呈现出一种不受欢迎的学习曲线,并减少了Java的可能采用。
简而言之,在Java中对对象(包括数组)进行参数传递的设计是专门服务于通过引用的语义意图,但是用以值传递引用的语法来实现。
有点诡计房地产…即使引用通过Java中的值传递,因此对引用本身的更改也在调用的函数级范围内。编译器和/或JVM通常会将值类型转换为引用。