今天翻看微信上有关Java技术的公众号时,看到了一篇关于Java中值传递的问题,文章讨论了在Java中调用函数进行传参的时候到底是值传递还是引用传递这个面试时会问到的问题。之前也接触过类似的问题,但只是知道是值传递,具体到为什么,一直不是太清楚。今天看了一下,算是明白了,写个博客记录一下。
首先先声明一下,在Java中函数传参是值传递,不是引用传递。要弄清楚这个问题之前要先弄清楚什么是值传递,什么是引用传递。
值传递(pass by value):是指在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对参数进行修改,将不会影响到实际参数。
引用传递(pass by reference):是指在调用函数时将实际参数的地址直接传递到函数中,这样在函数中如果对参数进行修改,将影响实际参数。
请注意我红色标记的字,很关键。相信很多人对于是值传递还是引用传递都会有这样一种认识:在传递基本数据类型的时,是值传递,在传递引用数据类型时是引用传递。原因不过就是像下面这样的代码所表现出来的。
package com.wuqi.p1; public class ValuePassTest { public static void main(String[] args) { int a = 1; //传递基本数据类型,因为是将a的值传递给param,所以即便在pass函数中改变了 //参数的值,a的值还是不会变。所以我们认为在传递基本数据类型的时候是值传递 pass(a); System.out.println("a= " + a); } private static void pass(int param) { param = 2; System.out.println("param= " + param); } }
package com.wuqi.p1; import com.wuqi.p2.User; public class PassTest2 { public static void main(String[] args) { User user = new User(); user.setName("wutianqi"); //传递对象,因为是将指向User的引用user传递给了param, //在函数中param.setName会反应到真实的对象中去。因此我们 //认为在这种情况下是引用传递 pass(user); System.out.println("my name is " + user.getName()); } private static void pass(User param) { param.setName("wuqi"); System.out.println("my name is " + param.getName()); } }
包括我自己以前也是这么认为的。但是我们都没有注意到这样一个问题。请看代码
package com.wuqi.p1; public class PassTest3 { public static void main(String[] args) { String name = "wutianqi"; //这里传递字符串参数,按照我们以前的观点这里应该传递的是将指向字符串的name引用 //传递给param,那么在pass函数中修改参数的值会直接影响到name引用指向的字符串 //的值,那么输出的结果依此应该是 my name is wuqi my name is wuqi pass(name); System.out.println("my name is " + name); } private static void pass(String param) { param = "wuqi"; System.out.println("my name is " + param); } }
这段代码按照我们对象是引用传递的思想,输出的结果就应该是代码中所说的那样。但是真实输出的结果确实下面这样
哎呦!什么情况!颠覆了你的认知?当初看到这段代码我也是大吃一惊!这说明之前的认识是错的!
别慌!让我们在看一下值传递和引用传递的概念吧。这里我标红的字体起作用了。引用传递是直接传递引用,那么在函数中对参数进行修改将会影响到实际参数。按照这个理论,那么毫无疑问,通过上面的例子,引用传递对于Java函数参数传递来说是错误的。在看看值传的概念,值传递是将实际参数复制一份,对参数的改变不会影响到实际参数。注意复制这两个字!!!在上面的例子中,如果我们认为是复制了name引用,也就是复制了name引用的值,然后传递给param。param="wuqi",其实相当于param=new String("wuqi"),这时param指向了一个新的对象。而实际参数name还是指向原来的对象。这样的话输出的结果和正确的就对上了。这样也就证明了在Java中是值传递而不是引用传递。
对于对象来说传递的是引用的一个副本给参数,这一点要铭记!