数组是通过值传递还是通过Java引用传递?

Are arrays passed by value or passed by reference in Java?

本问题已经有最佳答案,请猛点这里访问。

Possible Duplicate:
Is Java"pass-by-reference"?

数组不是Java中的原始类型,但它们也不是对象,所以它们是通过值传递还是通过引用传递?它是否依赖于数组包含的内容,例如引用或基元类型?


Everything in Java are passed-by value.。对于数组(它只是一个对象),数组引用按值传递。(就像对象引用是通过值传递的一样)。

将数组传递给其他方法时,实际上会复制对该数组的引用。

  • 通过该引用对数组内容所做的任何更改都将影响原始数组。
  • 但是,将引用更改为指向新数组不会更改原始方法中的现有引用。

看到这个帖子…

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
// assuming you allocated the list
public void changeArray(Integer[] list) {
    list = null;
}

如果传递一个非空列表,则在该方法返回时,它不会为空。


不,那是错误的。数组是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通常会将值类型转换为引用。