在搞懂这个问题之前, 我们要首先了解什么是值传递, 什么是引用传递?
- 值传递: 传递的是数据的副本,修改副本不会影响原始数据。
- 引用传递: 传递的是数据的引用(地址),修改引用会直接影响原始数据.
也就是说,值传递和引用传递最大的区别是传递的是自身,还是复制的副本,如果传递的是自身则为引用传递,如果传递的是复制的副本则为值传递。
但是在Java语言中, 只有值传递, 没有引用传递
下面我们看一个例子, 传递基本数据类型 int 的参数, 之后再修改传递的参数, 然后打印结果
package value_propagation;public class Test01 {public static void main(String[] args) {int a = 333;System.out.println("调用方法之前: " + a);change(a);System.out.println("调用方法之后: " + a);}public static void change(int number) {number = 666; // 修改 number 的值}
}
看上面结果可以知道 int 是值传递, 传递的只是个副本, 并没有修改原始值
然后我们传递一个对象(引用类型) 试试
package value_propagation;public class Test02 {public static void main(String[] args) {Integer a = new Integer(333);System.out.println("调用方法之前: " + a);change(a); // change(a) 传递的是 a 引用的副本System.out.println("调用方法之后: " + a);}public static void change(Integer number) {number = 666; // 修改 number 的值,}
}
在 change 方法中,number 是局部变量,它接收到 a 引用的副本。 因此,当你修改 number 为 666 时,实际上只是改变了 number 这个局部变量的引用指向的内容。这并不会影响外部的 a,因为 a 和 number 是两个不同的引用。 a 仍然指向 Integer 对象 333,而 number 只是一个局部的副本。
下面我们看一个有争议的传递, 传递一个数组
package value_propagation;public class Test03 {public static void main(String[] args) {int[] arr = {1, 2, 3, 4};System.out.print("调用方法之前: ");for (int i = 0; i < arr.length; i++) {System.out.print(arr[i] + " ");}change(arr);System.out.print("\n调用方法之后: ");for (int i = 0; i < arr.length; i++) {System.out.print(arr[i] + " ");}}public static void change(int[] array) {array[2] = 666;}
}
这是不是说明Java支持引用传递呢?
并不是。这里在参数传递的过程中,只是把实参的地址拷贝了一份传递给形参. update方法中只修改了参数地址里面的内容,并没有对形参本身进行修改。
下面我们将代码稍加改动
package value_propagation;public class Test04 {public static void main(String[] args) {int[] arr = {1, 2, 3, 4};System.out.print("调用方法之前: ");for (int i = 0; i < arr.length; i++) {System.out.print(arr[i] + " ");}change(arr);System.out.print("\n调用方法之后: ");for (int i = 0; i < arr.length; i++) {System.out.print(arr[i] + " ");}}public static void change(int[] array) {array = new int[] {11,22,33,44};}
}
如果是引用传递,那么我在新方法中无论如何修改,那么都应该是影响原对象才对,而刚才我稍微调整了代码之后就发现其并非引用传递,而是值传递,这是因为当传递数组时,其传递的是"引用副本”,而非真正的引用对象(也就是其本身)。
也就说,当传递数组时,其实传递的是“引用副本”, 还是值传递