Java 到底是值传递还是引用传递?
Java 到底是值传递还是引用传递?
首先我们先了解下值传递和引用传递的基本概念:
值传递(pass by value)是指在调用函数时将实际参数复制一份传递到函数中, 这样在函数中如果对参数进行修改,将不会影响到实际参数。 引用传递(pass by reference)是指在调用函数时将实际参数的地址直接传递到 函数中,那么在函数中对参数所进行的修改,将影响到实际参数。
两种方式根本区别就在于是否创建副本,所导致结果的区别为是否会直接操作原始对象。
首先基本类型 :
package com.juc.demo.test; public class aTest { static void change(int value) { value = 100; } public static void main(String[] args) { int num=2; change(num); System.out.println("after change num = " +num); } }
输出结果 为2,没有改变原始值,不管我们在change()方法里面对value如何操作,都不会改变num的值,很明显是值传递。
after change num = 2
对于引用类型来说,我们通过两个例子来说明:
第一个例子:
package com.juc.demo.test; public class aTest { static void change(User user) { user.setName("李四同学"); } public static void main(String[] args) { User user= new User(); user.setName("张三同学"); change(user); System.out.println("after change name = " +user.getName()); } } class User{ private String name; public String getName() { return name; } public void setName(String name) { this.name = name; } }
after change name = 李四同学
第二个例子:
package com.juc.demo.test; public class aTest { static void change(User user) { user=new User(); user.setName("李四同学"); } public static void main(String[] args) { User user= new User(); user.setName("张三同学"); change(user); System.out.println("after change name = " +user.getName()); } } class User{ private String name; public String getName() { return name; } public void setName(String name) { this.name = name; } }
after change name = 张三同学
我们可以看到,第一个例子结果改变了原始对象的name,第二个没有改变,为什么呢? 假设原user对象的地址是0x123456,其实,不管是哪个例子,java都有复制一个新的副本,保存的是user对象的地址0x123456,我们把这个复制的地址传进change()方法里。
-
第一个例子,虽然创建了引用地址的副本,由于该地址副本指向的仍是堆中的user对象,所以对该对象操作会被改变; 第二个例子中,user=new User(); 开辟了新的堆空间,假设地址是0x567899,使得副本指向该地址,然后再对该地址的user进行操作,也不会影响到原user对象。
如果java是引用传递,即直接将该地址传递过去,而没有拷贝新的副本,那第二个例子中参数会指向新的地址,对新地址的user操作,结果一定会改变。
所以,不管是基本类型的数据,还是引用类型,java都是拷贝了新的副本传值,只不过基本数据类型拷贝的是数据本身,而引用类型拷贝的是引用地址,并非直接把该地址传进去。两种方式根本区别就在于是否创建副本,而不是说传递的类型是值还是引用。 总而言之java只有值传递,不存在引用传递。
上一篇:
IDEA上Java项目控制台中文乱码