当一个对象被当作参数传递到一个方法后,在此方法内可以改变这个对象的属性,那么这里到底是「按值传递」还是「按引用传递」?
答:是按值传递。Java 语言的参数传递只有「按值传递」。当一个实例对象作为参数被传递到方法中时,参数的值就是该对象的引用的一个副本。指向同一个对象,对象的内容可以在被调用的方法内改变,但对象的引用(不是引用的副本) 是永远不会改变的。
public class Test {
public static void main(String[] args) {
Person person= new Person();
person.name = "hello";
int num = 1;
invosky(person,num );
System.out.println("属性:"+num );
System.out.println("对象:"+person.name);
}
public static void invosky(Person p,Integer x){
p.name = "Whate";
x= x+ 4;
}
}
class Person{
public String name;
}
输出结果:
为什么会出现这种情况, 主要是基本类型参数和对象类型参数传递没有搞清除。
基本类型参数传递图:
num作为参数传递给invosky()方法时,是将内存空间中num所指向的那个存储单元中存放的值1传递给了invosky()方法中的x变量,而这个x变量也在内存空间中分配了一个存储单元,这个时候,就把num的值1传递给了x的这个存储单元中。此后,在invosky()方法中对x的一切操作都是针对x所指向的这个存储单元,与num所指向的那个存储单元没有关系了!
所以,在invosky()方法调用之后,num所指向的存储单元的值还是没有发生变化,这就是所谓的“按值传递”!按值传递的精髓是:传递的是存储单元中的内容,而不是存储单元的引用!
对象的传递过程:
程序在堆内存中开辟了一块内存空间用来存储Person类的实例对象,同时在栈内存中开辟了一个存储单元用来存储该实例对象的引用,即上图中person指向的存储单元。
person作为参数传递给invosky()方法,需要注意的是:person将自己存储单元的内容传递给了invosky()方法的p变量!此后,在invosky()方法中对p的一切操作都是针对p所指向的存储单元,与person所指向的那个存储单元没有关系了!
p中name改变,利用引用直接修改了堆中对象的值,从而输出person中name的值也发生了改变。