java 是值传递(pass by value),还是引用传递(pass by reference)?
答案是:值传递(pass by value)
先说一下值传递和引用传递的概念和区别吧
值传递:是指在调用函数时,将原始参对象复制一份作为实参传给形参,当在函数中对参数进行修改时,不会影响到实际参数。
引用传递:是指在调用函数时,将原始对象直接作为实参传给形参,一旦在函数中对其进行修改,将会影响实际参数。
其主要区别是:
接下来让我们看一段例子:
public static void main(String[] args) {
int i = 2;
modify(i);
System.out.println("print in main : i = " + i);
}
public static void modify(int j) {
j = 4;
System.out.println("print in modify : j = " + j);
}
打印结果如下:
print in modify : j = 4
print in main : i = 2
从打印结果可以看出,当我们修改了形参j时,对于实际参数i,并没有发生改变,这替我们证明了,对于基本数据类型来说,传递的是值
接下来让我们看看引用数据类型
public static void main(String[] args) {
A a = new A();
a.setName("Martina");
modify(a);
System.out.println("print in main : a.name = " + a.getName());
}
public static void modify(A b) {
b.setName("Leon");
System.out.println("print in modify : b.name = " + b.getName());
}
打印结果如下:
print in modify : b.name = Leon
print in main : a.name = Leon
我们发现,实际参数a的内容随着函数里的修改也发生了改变,这是不是说明对于引用类型,是引用传递呢,不,希望你们不要产生这样的误解。对于引用类型,依旧是值传递,先看看下面一段代码。
public static void main(String[] args) {
A a = new A();
a.setName("Martina");
modify(a);
System.out.println("print in main : a.name = " + a.getName());
}
public static void modify(A b) {
b = new A();
b.setName("Leon");
System.out.println("print in modify : b.name = " + b.getName());
}
看看打印结果:
print in modify : b.name = Leon
print in main : a.name = Martina
这个时候我们的原始对象并没有发生改变,而这一段与上一段的区别在于:
b = new A();
如果是引用传递,那么我们的原始参数a应该还是会发生影响的,所以还是先让我们看一下这两个例子到底发生了什么:
当我们在传递参数时,将原始对象的地址传给了形参,在函数中,对形参的内容进行了改变,也就是,改变了原始对象的内容。
而在 b=new A(); 的时候,改变了形参的地址,对原始对象的内容没有进行影响。故而我们可以判断出,这并不是引用传递,如果是,在 b=newA(); 的时候,a的地址也应该发生改变,事实并没有,故而证明java就是值传递。
还有一个比较生动的例子来描述值传递和引用传递。
小剧场:
小明家有一个电视,有一天,他的合租室友小吴想看电视了,小明就去店里给他配了一个遥控器。这一天,小明和小吴,在家里看电视,小吴在小明去倒水的时候说,我们看“CCTV”吧,于是他用自己的遥控器换了台,等小明回来,他看到的就是CCTV正在播放的电影频道。忽然小吴不小心把水洒在了自己的遥控器上,于是他就要用小明的遥控器换台了,他一不小心将手里的番茄酱沾在了小明的遥控器上,等小明拿到遥控器的时候,沾了一手的番茄酱。
你有一个遥控器,专门操作这个电视,值传递就是,你复制了一个遥控器给了别人,他也可以操作这个电视,当他换台了,也就意味着这个电视机的内容发生了改变,但是你的遥控器他却没有改变的权利,引用传递就是,你把自己的遥控器给了别人,别人拿着你的遥控器对电视机进行操作,他还可以对你的遥控器进行操作,这便是引用传递。故而由电视的内容是否改变来判断这个问题是不太适合的,咱也不要纠结为啥内容发生了改变,咱给的是遥控器啊,看的是遥控器有没有变。b=new A();这个操作则是给了他另一个电视的遥控器。
总结
让我们再来梳理清楚,java是值传递这个概念,我们在上面已经给出了值传递和引用传递之前的区别:是否复制副本,是否会影响到原始对象。根据上面的例子,我们可以总结:对于基本数据类型,传递的是原始数据的值的副本,对于引用数据类型,传递的是原始数据地址值的副本。
看看人外国大佬的回答:
所以,别问,问,就是值传递!!!