Java值传递 浅谈Java到底是值传递还是引用传递呢
XH雪浪风尘 人气:0一、前言
最近在看Java核心卷一,也就是这本书:
在这本书里面也看到了这个问题,Java是值传递还是引用传递,这个问题其实也是很有意思的,之前也看到过这个问题,但是只是依稀记得是值传递,而且网上也有在讨论这个问题的。所以就先说结论吧:是值传递。
二、值传递与引用传递
既然讨论是值传递还是引用传递,那肯定是要知道啥是值传递、引用传递的。
值传递:是指在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对参数进行修改,将不会影响到实际参数。
引用传递:是指在调用函数时将实际参数的地址直接传递到函数中,那么在函数中对参数所进行的修改,将影响到实际参数。
所以我们就可以做个简单的比较:
值传递 | 引用传递 | |
将参数复制一份传递过去 | 将参数的实际地址传递过去 | |
不会影响到实际参数 | 会影响到实际参数 |
我们在方法中,传递参数类型有两种:基本数据类型(数字、布尔)以及对象引用这两种,所以我们就从这两种类型进行分析。
三、基本数据类型
以数字int为例:
public class Test { public static void main(String[] args) { Test test=new Test(); int i=1; test.incr(i); System.out.println("main中i的值大小为"+i); } public void incr(int i){ i=i+1; System.out.println("incr中i为"+i); } }
main方法中的结果要么为1,要么为2,如果还是1的话,那么很大可能就是值传递,我们看下输出结果:
incr中i为2
main中i的值大小为1
可以看到,main方法中的值仍为1,我们来看下这个的过程是怎么样的:
incr方法中虽然对i的值进行了加一操作,但是他只是将值复制了一份,incr方法执行完毕之后,就会被处理掉,并没有改掉原先的值,所以才会在main方法中打印出i还是原先的值。
四、对象引用
基本数据类型其实比较好解释,对象引用其实还是有那么一点迷惑性的,因为有的人可以认为对象引用是引用传递,他可以向方法中传递一个对象,然后在子方法中修改对象的值,就比如下面的这个例子:
例子一:
public class Test { public static void main(String[] args) { Test test=new Test(); Student s1=new Student(); s1.setId(1); test.changeId(s1); System.out.println("main中的s1id是"+s1.getId()); } public void changeId(Student student){ student.setId(2); System.out.println("changeId中的id为"+student.getId()); } } class Student{ private int id; public int getId() { return id; } public void setId(int id) { this.id = id; } }
一个很简单的例子,创建个student对象,只有id这一个字段,将student对象传递到子方法中,执行结果会是什么呢?
changeId中的id为2
main中的s1id是2
可以看到阿,main方法与changeId方法中,他们最后的id都是2,所以有的人就会认为,修改了对象中的值,所以是值传递。
其实我们看上面值传递与引用传递的概念,引用传递是传递的地址,那么我们就假设对象引用是引用传递方式,那么我传递进去两个对象,交换他们的位置是可以改变他们的指向的,接下来我们就来看一下会不会改变:
例子二:
package com.dong.No2; public class Test { public static void main(String[] args) { Test test=new Test(); Student s1=new Student(); Student s2=new Student(); s1.setId(1); s2.setId(2); test.changeId2(s1,s2); System.out.println("main中的s1id是"+s1.getId()+","+"s2id是"+s2.getId()); } public void changeId2(Student s1,Student s2){ Student s3=s1; s1=s2; s2=s3; System.out.println("changeId2中的s1id是"+s1.getId()+",s2id是"+s2.getId()); } } class Student{ private int id; public int getId() { return id; } public void setId(int id) { this.id = id; } }
他的执行结果:
changeId2中的s1id是2,s2id是1
main中的s1id是1,s2id是2
我们发现,这个在子方法中,对象的值是改变了,但是main方法中的值还是原先的样子,那这样就不符合引用传递了,因为他其实并没有改变原先的对象。
如果我们以值传递的观点来解释,那么就可以说的通了,我们传入的两个参数s1、s2在传递过去后,会复制一份为s1复制、s2复制,然后在子方法中,是对这两个复制过后的对象进行的操作,执行完之后,这些复制的对象就会被回收,所以就出现了我们在主方法中,看到这两个对象的值是没有改变的。
事实上也是如此,我们可以同样来解释例子一种,为啥传入了对象,但是主方法中的值却改变了。
我们知道对象这些都是在堆中存储的,我们在向方法中传递的,实际上是这个对象在堆中的地址,我们传递的对象,实际上就是传递的对象的地址:
因为s1与s1复制都是指向的ox123456,s1复制改变了值,那么s1看到的值也就发生了改变,即使是s1复制最后被回收,ox123456的改变不会恢复。
五、结论
所以Java中的传递只有值传递而没有引用传递,只不过传递为基本数据类型的话,是复制的数值,而对象类型的话,则是复制的对象存放地址。
加载全部内容