1、什么是值传递
值传递,是将内存空间中某个存储单元中存放的值,传送给还有一个存储单元。(java中的存储单元并不是物理内存的地址,但具有相关性)
比如:
//定义了一个改变參数值的函数
public static void changeValue(int x) {
x = x *2;
}
public class TestMain{
//调用该函数
int num = 5;
System.out.println(num);
changeValue(num);
System.out.println(num);
}
结果例如以下:
5
5
调用函数changeValue()前后num的值都没有改变。详细过程如图:
- num作为參数传递给changeValue(int x)方法时。首先在内存空间中为x变量分配一个存储单元(我们说x指向这个存储单元)。
- 将内存空间中num指向的存储单元中存放的值(即”5”),传递给了changeValue(int x)中的參数变量(即”x”),也就是把”5”传给了x变量指向的存储单元中。
- changeValue(int x)方法中对x变量的一切操作。都是针对x指向的存储单元。与num指向的存储单元没有关系,当然也不会改变这个存储单元中的值。
所以。值传递,传递的是存储单元中的内容(8种基本类型:值,非基本类型:实际对象的地址)。
对于String来说JVM有他特殊的处理,了解很多其它看这里:String在内存中怎样存放
2、什么是引用传递
java中仅仅有值传递,没有引用传递。
所谓的引用传递,仅仅是一个错误的概念。
比如:
class person {
public static String name = "Jack";
//定义一个改变对象属性的方法
public static void changeName(Person p) {
p.name = "Rose";
}
public static void main(String[] args) {
//定义一个Person对象,person是这个对象的引用
Person person = new Person();
//先显示这个对象的name属性
System.out.println(person.name);
//调用changeName(Person p)方法
changeName(person);
//再显示这个对象的name属性。看是否发生了变化
System.out.println(person.name);
}
}
运行后结果:
Jack
Rose
从结果看,方法用了一个对象參数。操作參数就能够改变传入对象。我们的对引用传递的错误观念这么觉得:
该对象复制了一个引用副本,传给调用方法的參数。使得该方法能够对这个对象进行操作。
这样的观念是刚開始学习的人常犯的错误。
实际上过程如图:
- main方法中new了一个对象Person,存储空间中实际分配了两个对象:新创建Person类的实体对象、指向该对象的引用变量person。
当中,实体对象存放在堆内存中,引用变量存放在栈内存(Java存储特性)。
了解很多其它java存储看这里: - 引用变量person指向的栈内存中。存放的是堆中实体对象的逻辑地址。
- 调用changeName(Person p)方法,将person引用变量传入该方法參数p中(依照值传递,传递的是:实体对象的逻辑地址)。
此时。changeName方法中对p的操作。与person没有关系。
- changeName方法中。是对p指向的存储单元中的值(即实体对象的逻辑地址)所指向的实体对象进行操作。直接改变了该实体对象。
- 因为person指向的存储单元中的值也是该实体对象的逻辑地址。这个实体对象已经在第4步中被改变了。
所以有上面的结果。
3、引用传递和值传递有什么差别
引用传递是个伪概念。java中仅仅有值传递。