文章目的:验证Java语言到底是值传递还是引用传递以及Java参数传递的实现原理.
问题引入:
先阅读代码段:
public static void main(String[] args){
Person p=new Person("张三");
f(p);
System.out.println("实参:"+p);
}
public static void f(Person p){
p.name="李四";
System.out.println("形参:"+p);
}
运行结果:
形参:Person{name=“李四”}
实参:Person{name=“李四”}
我们将一个对象变量传递到方法中,然后在方法内修改对象的属性,打印实参和形参,从表面上看,实参形参的值都被修改了.难道Java也支持引用传递吗?
答案是:Java语言只支持值传递,不支持引用传递. |
在验证Java参数传递方式之前,我们先要明白函数值传递和引用传递的区别.
可以看出值传递和引用传递的区别在于对形参的修改会不会影响到实参
我们将一个基本数据类型传递到方法中,然后在方法中修改形参的值,发现形参的修改并没有影响到实参.
public static void main(String[] args){
int a=1;
f(p);
System.out.println("实参:"+a);
}
public static void f(int a){
a=2;
System.out.println("形参:"+a);
}
在问题引入的例子中,看似形参的修改影响到了实参,但是只要我们创建一个新对象,并赋值给形参,此时再打印实参和形参,两者互不影响.说明Java并不符合引用传递.
public static void main(String[] args){
Person p=new Person("张三");
f(p);
System.out.println("实参:"+p);
}
public static void f(Person p){
p=new Person("李四");
System.out.println("形参:"+p);
}
我们要明白为什么会发生这些现象,就要理解Java背后的基本原理:
JVM划分了很多块区域,我们创建的对象就放在堆中,而基本数据类型和局部变量都放在栈中.当传递基本数据类型时,是将数据创建了一个副本传递到方法中,所以实参不会受到形参修改的影响,如图:
由于对象是放在堆区的,所以我们只能使用对象的引用来操作这个对象.
当对象引用传递给方法时,其实是创建了一个引用副本,同时指向同一个对象,通过形参引用操作对象时,就好像实参发生了变化,其实对象内容改变了,但是对象变量没有改变,实参本身是没有改变的.因此引入问题中的例子就不难理解了.
当我们将形参重新赋值的时候,实参不会受到任何影响,此时,实参和形参已经指向了两个不同的对象.
所以说,Java只支持值传递.