Java值传递:
是针对Java基本类型(int等)数据的传递,在方法执行完以后,形参值的改变不会影响到实参的数据,这个没什么好说的。
Java引用传递:
首先需要明确一点,引用的含义,引用是个通用名词,在C++中表示变量的别名,而在Java中,引用类似于C++的指针,但又不太相同,因为Java的引用并不需要你自己去控制对象的使用周期,由于Jvm自带垃圾回收,所以该引用是是对象在jvm中的内存地址,Java为了避免出现内存泄露问题,出于安全性以及GC回收机制,也不允许对该变量进行自增操作,而是让Java程序直接使用该内存地址的内容,C++中的指针是可以的。
所以有很多人说,Java并没有引用传递,也是对的,只不过他们把此处的引用当做了C++的引用传递而已。
我们姑且在Java中传递地址的方式称为引用传递,那么Java到底是不是传递的地址呢,看示例。
public class User {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
public class Test {
public void set(User user){
user.setName("hello world");
}
public static void main(String[] args) {
Test test = new Test();
User user = new User();
test.set(user);
System.out.println(user.getName());
}
}
上面代码的输出结果是“hello world”,这不必多说,那如果将set方法改为如下,结果会是多少呢?
public void set(User user){
user.setName("hello world");
user = new User();
user.setName("change");
}
答案依然是“hello world”,下面就让我们来分析一下如上代码。
首先
User user = new User();
是在堆中创建了一个对象,并在栈中创建了一个引用,此引用指向该对象,如下图:
test.set(user);
是将引用user作为参数传递到set方法,注意:这里传递的并不是引用本身,而是一个引用的拷贝。也就是说这时有两个引用(引用和引用的拷贝)同时指向堆中的对象,如下图:
user = new User();
在set()方法中,又创建了一个User对象,并将“user引用的拷贝”指向这个在堆中新创建的对象,如下图:
user.setName("change");
在set()方法中,“user引用的拷贝”操作的是堆中新创建的User对象。
System.out.println(user.getName());
set()方法执行完毕,目光再回到mian()方法,因为之前,”user引用的拷贝”已经将堆中的User对象的name属性设置为了”hello world”,所以当main()方法中的user调用getName()时,打印的结果就是”hello world”。如下图:
Java虚拟机体系结构