java到底是引用传递还是值传递?

时间:2021-12-23 16:41:42

今天我们来讲讲一个在学习中容易误解的问题,面试中也偶尔问到,java方法调用时到底是值传递还是引用传递?

首先,请大家来做一个判断题,下面的3个问题是否描述正确

1. java基本数据类型传递是值传递,引用数据类型是引用传递

2. java都是引用传递

3. 方法调用时传的是数据值就是指传递,传的是地址就是引用传递

我们暂且不直说上面3个问题的正确性,下面,我们通过几个测试案例来测试一下参数传递

1. 基本数据类型当作参数

public class test1 {
public static void main(String[] args) throws Exception {
int a = 10;
change(a);
System.out.println("a = " + a); }
public static void change(int s) {
s = 20;
System.out.println("s = " + s);
}
}

 运行结果:  方法执行后,没有改变原有a的值

s = 20
    a = 10

2. 引用类型当作参数

public class Student {
private String name; //姓名
Student(String name){
this.name = name;
} public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
} } public class Test1{

   public static void main(String[] args) throws Exception {
      Student s = new Student("haly");
      change(s);
      System.out.println("s = " + s.getName());

  }

public static void change(Student x) {
         x.setName("wuwu");
         System.out.println("x = " + x.getName());
   }

}

运行结果: 方法执行后,改变了原来s的值

x = wuwu
s = wuwu

看到这里,很多同学就觉得问题1答案是正确的, java基本数据类型传递是值传递,引用数据类型是引用传递,但是真的是这样吗?

这里我可以提前透露下答案,上面这个测试之所以改变了原来实参s里面的值,是因为我们传递到方法的都是实参对象的一个拷贝,x对象是s对象引用的一个拷贝,

所以x和s都是指向同一块内存地址,改变x里面的值,相当于改变s的值,因为传的是对象引用的拷贝,而不是对象引用本身,所以是值传递。

如果不好理解,我给大家举个例子:

我有一辆车,我自己有一把车钥匙,有一天我的女朋友想要开车,我就配制了一把车钥匙给她,这时候,相当于方法调用,传递了一把钥匙的拷贝给她。

她在钥匙上刻字,对我的钥匙是没有影响的,但是她拿钥匙打开车,将我的车坐垫换了,我开这辆车的时候,车坐垫的确是换掉了,这是同一个道理。

如果还不清楚,我用两张图,分别表示案例1和案例2的值传递:

java到底是引用传递还是值传递?                            java到底是引用传递还是值传递?

案例1                                                                                               案例2

如果大家还不好理解,下面我就写一个案例3,让大家更加清晰的看出,java是值传递

3. 引用类型当作参数

public class Test1 {
public static void main(String[] args) throws Exception {
Student s1 = new Student("小张");
Student s2 = new Student("小李");
Test1.swap(s1, s2);
System.out.println("s1:" + s1.getName() + ",s2:" + s2.getName()); }
public static void swap(Student x, Student y) {
Student temp = x;
x = y;
y = temp;
System.out.println("x:" + x.getName() + ",y:" + y.getName());
} }

运行结果:

x:小李,y:小张
s1:小张,s2:小李

通过测试3的代码,我们可以看出,在方法中交换了两个参数的值,但是原来的值并没有交换,如果是引用传递的话,我们操作的都是实参本身的引用,按理也会交换的。

下面,我们通过一个图在仔细思考一下答案:我们交换了x和y对象的引用,对原来的s1和s2并没有影响。

java到底是引用传递还是值传递?

通过以上代码及图描述,我们可以得出如下2个结论:

1. 如果我们传递的实参是对象引用的拷贝,就是值传递,如果我们传递的实参是引用本身,就是引用传递。

2. 在java中,不管基本数据类型,还是引用类型,都是值传递。

如有不正确的地方,欢迎大家批评指出,一起进步~~