先看一段代码:
public static void add(String a) {
a = "new";
System.out.println("add: " + a); // 输出内容:add: new
}
public static void main(String[] args) {
String a = null;
add(a);
System.out.println("main: " + a); // 输出内容:main: null
}
我们知道一个基本的概念:在java中基本类型是按值传递,而引用类型是地址传递。基于这个概念那么main方法中的a最后应该输出‘new’才对呀,为什么还是:‘null’呢?要理解这个问题,我们引入一个概念,在进行方法调用的时候,形参a和实参a的地址其实是不一样的,发生了一次地址copy。
add方法调用前的堆栈状态如图所示:
执行完add方法后的堆栈状态,如图所示:
可以看出来,进行方法调用的时候形参a和实参a的地址发生了一次地址复制,一开始都是指向null,但是add方法执行完后,其中add方法中参数a地址重新指向了字符串“new”,并没有改变main方法中参数a的指向,因此,不论add方法调用多少次,main方法中参数a的值都没有发生变化。
为什么方法调用会进行一次地址复制呢?
方法调用的本质其实就是一次入栈,调用结束后进行出栈操作。入栈就是要保护当前上下文环境,出栈就是要恢复到调用方法前的状态。为了保护上下文环境就导致必须对参数进行一次复制,对于基本类型是值复制,对于引用类型就是地址值的复制。如果还是拿着之前的参数进行操作就没法恢复上下文环境了。
结论
Java对于引用类型的参数虽然是引用传递,但是要注意,方法调用会导致实参和形参的地址值发生一次复制,从而导致对形参进行new赋值时,导致形参和实参指向不一致,导致潜在的问题。
建议
尽量不要对方法参数进行new重新赋值操作,从而导致潜在的问题。如果非要这么做,请先弄清楚这样做后的后果,从而避免潜在的bug。