问题
关于Java传参时是引用传递还是值传递,一直是一个讨论比较多的话题,有论坛说Java中只有值传递,也有些地方说引用传递和值传递都存在,比较容易让人迷惑。
答案
但是请记住,Java 语言的参数传递只有「按值传递」
Java的参数传递,不管是基本数据类型还是引用类型的参数,都是按值传递,没有按引用传递!
先看一下microsoft的文档中对按引用传递参数的定义(如下截图)(看不懂的话就先看下面的,回头再看这个,你会有一种醍醐灌顶的感觉):
-
基本数据类型的参数
先来看一下基本数据类型的参数按值传递的例子:
TransferTest.java
public class Test{
public static void main(String[] args) {
int num = 1;
System.out.println("changeNum()方法调用之前:num = " + num);
changeNum(num);
System.out.println("changeNum()方法调用之后:num = " + num);
}
public static void changeNum(int x) {
x = 2;
}
}
运行结果:
这个传递过程的示意图如下:
num作为参数传递给changeNum()方法,是将内存空间中num所指向的存储单元中存放的1传递给了changeNum()方法中的x变量。调用changeNum(),x变量在内存空间中分配一个存储单元,这个时候,num的值1传递给了x指向的存储单元中。此后,在changeNum()方法中对x的一切操作都是针对x所指向的这个存储单元,与num所指向的那个存储单元没有关系了!
所以,在changeNum()方法调用之后,num所指向的存储单元的值还是没有发生变化,这就是所谓的“按值传递”!按值传递的精髓是:传递的是存储单元中的内容,而不是存储单元的引用!
-
引用类型的参数
Test2.java
1.public class Test2 {
2. public static void main(String[] args) {
3. Person p1 = new Person();
4. System.out.println(p1);
5. change(p1);
6. System.out.println(p1);
7. }
8.
9. public static void change(Person p2) {
10. System.out.println(p2);
11. p2 = new People();
12. System.out.println(p2);
13. }
14.}
15.
16./**
17. * Person类
18. */
19.class Person {
20.
21.}
22.
运行结果:
可以看出四次打印person的地址值,只有new之后的变量地址值不一样,并且用完change() 方法之后,person变量并没有发生改变。
这个传递过程的示意图如下:
当执行到第3行代码时,程序在堆内存中开辟了一块内存空间用来存储Person类的实例对象,同时在栈内存中开辟了一个存储单元用来存储该实例对象的引用,即上图中person指向的存储单元。
当执行到第5行代码时,person作为参数传递给change()方法,需要注意的是:person将自己存储单元的内容传递给了change()方法的p变量(即引用对象的地址)!此后,在change()方法中对p的一切操作都是针对p所指向的存储单元,与person所指向的那个存储单元没有关系了!如11行,对p变量指向了新的对象,这个传递过程如下图
总结
-
Java 语言的参数传递只有「按值传递」
-
基本类型传参时,传递给形参的就是的实参的值
-
引用类型传参时,传递给形参的是实参的引用对象的值