最近一直疑惑一个问题,Java函数传递的形参会影响实参的值吗,听到网上很多说法,有的说java只有值传递,有的说java有值传递和引用传递,很迷糊。其实没有这么复杂,值传递、址传递、引用传递这些是c/c++中的概念,Java中甚至不用考虑这些专属名词。
比较重要的就是了解JVM的内存机制,基本数据类型是存放到栈中的,引用数据类型是存在栈中的引用,指向存在堆中的实际对象(引用就是堆中对象的地址)。
1.基本数据类型作为参数传递
参数传递基本数据类型时,形参不会改变实参,因为不同的函数在栈中存放的地址不同,形参是栈中对实参的拷贝,所以不会改变值。这个比较简单就不谈了。
直接上代码进行测试:
public class text {
public static void main(String[] args) {
int num = 1;
System.out.println("changNum方法调用之前的num值"+num);
changeNum(num);
System.out.println("changNum方法调用之后的num值"+num);
}
public static void changeNum (int num){
num = 2;
System.out.println("changNum方法中的num值"+num);
}
}
运行结果为:
changNum方法调用之前的num值1
changNum方法中的num值2
changNum方法调用之后的num值1
显然实参和形参是两个不一样的东西,而且方法内形参的改变没有影响实参。
如图所示:基本数据类型是不存放在堆里的,实参传递给形参是弄一个新的副本给changeNum函数所在栈的区域,两个不同栈地址的num当然不会相互影响
2.引用数据类型作为参数传递
上面说到了引用数据类型是栈里的一个引用指向堆中的地址,我们以字符串为例子:
String abc = new String(“abc”);
在内存中的情况大概就是:
引用类型作为参数传递时,因为是栈中的两个方法,所以传递的引用也处于栈的不同位置,但是他们记录的堆中的地址是同一个,上代码:
public class text {
public static void main(String[] args) {
User user = new User();
user.age = 1;
System.out.println("change方法调用前user的age值为"+user.age);
change(user);
System.out.println("change方法调用后user的age值为"+user.age);
}
public static void change (User user){
user.age =2;
System.out.println("change方法中user的age值为"+user.age);
}
}
class User{
public int age;
User(){
}
User(int a){
this.age =a;
}
}
运行结果为:
change方法调用前user的age值为1
change方法中user的age值为2
change方法调用后user的age值为2
此时user对象中的age已经变了,内存情况大概是这样:
因为两个引用指向的是一个堆中的对象,所以当然值就发生改变了。
如果此时,在change方法中,把形参这个引用再指向一个新new的User对象呢,能影响实参的User吗?
public static void change (User user){
user = new User(2)
System.out.println("change方法中user的age值为"+user.age);
}
运行结果:
change方法调用前user的age值为1
change方法中user的age值为2
change方法调用后user的age值为1
因为此时指向的堆中的对象,已经不是同一个了:
所以当然没有影响实参
最后声明下,图里的地址都是我瞎编的哈,知道他们不一样就行了。