观点:Java没有引用传递,只有值传递
基本概念
- 实参:实际参数,是提前准备好并赋值完成的变量。分配到栈上。如果是基本类型直接分配到栈上,如果是引用类型,栈上分配引用空间存储指向堆上分配的对象本身的指针。String等基本类型的封装类型比较特殊,后续讨论。
- 形参:形式参数,方法调用时在栈上分配的实参的拷贝。
- 值传递:方法调用时,实际参数把它的值传递给对应的形式参数,形参接收的是原始值的一个拷贝,此时内存中存在两个相等的变量
- 引用传递:方法调用时将实参的地址传递给对应的形参,实参和形参指向相同的内容
Java数据有两个类型
- 基本类型
- 引用类型
![](https://img.laitimes.com/img/9ZDMuAjOiMmIsIjOiQnIsIyZuBnL1cDN3QzM5QzN5YWO3UjNhljNlhzM4kzN2MjM3kjMhZmNjF2YkVmMx8CXt92Yu4GZjlGbh5SZslmZxl3Lc9CX6MHc0RHaiojIsJye.png)
如上图所示,基本类型传递时,线程在栈上分配形式参数并拷贝实际参数的值。
/** * Create with IntelliJ IDEA * Author : wangzhenpeng * Date : 2018/4/8 * Time : 上午9:48 * E-mail : [email protected] * Description : Test */ public class Test { public static void main(String[] args) { int value = 100; change(value); System.out.println("outer: " + value); } static void change(int value) { value = 200; System.out.println("inner: " + value); } }
结果输出:
inner: 200 outer: 100
方法修改的只是形式参数,对实际参数没有作用。方法调用结束后形式参数随着栈帧回收。
如上图所示,引用类型传递时,传递的是引用的值,从这个角度来讲还是值传递。如果是引用传递的话,传递的应该是引用的地址,而不是引用的值。
/** * Create with IntelliJ IDEA * Author : wangzhenpeng * Date : 2018/4/8 * Time : 上午9:48 * E-mail : [email protected] * Description : Test */ public class Test { public static void main(String[] args) { Person person = new Person(); person.setAge(17); person.setName("Tom"); change(person); System.out.println("outer: " + person.getAge()); } static void change(Person value) { value.setAge(18); System.out.println("inner: " + value.getAge()); } static class Person { private String name; private Integer age; public String getName() { return name; } public void setName(String name) { this.name = name; } public Integer getAge() { return age; } public void setAge(Integer age) { this.age = age; } } }
inner: 18 outer: 18
方法修改的是引用所指向的数据空间的数据,所以方法外部也能看到修改的结果。
基本类型的数组也是对象,所以int[] 传递的也是对象应用的值。
/** * Create with IntelliJ IDEA * Author : wangzhenpeng * Date : 2018/4/8 * Time : 上午9:48 * E-mail : [email protected] * Description : Test */ public class Test { public static void main(String[] args) { int[] intArray = new int[10]; change(intArray); System.out.println("outer: " + intArray[0]); } static void change(int[] value) { value[0] = 200; System.out.println("inner: " + value[0]); } }
运行结果:
inner: 200 outer: 200
如果对引用类型的传递稍作修改
static void change(Person value) { value = new Person(); value.setAge(18); System.out.println("inner: " + value.getAge()); }
inner: 18 outer: 17
用图解释为:
同理String,Integer等类型的封装类型为final类型,对数据的修改操作实际上是创建了一个新的对象
/** * Create with IntelliJ IDEA * Author : wangzhenpeng * Date : 2018/4/8 * Time : 上午9:48 * E-mail : [email protected] * Description : Test */ public class Test { public static void main(String[] args) { String value = "123"; change(value); System.out.println("outer: " + value); } static void change(String value) { value = "abc"; System.out.println("inner: " + value); } }
inner: abc outer: 123