方法调用是编程语言中非常重要的一个特性,在方法调用时,通常需要传递一些参数来完成特定的功能。Java语言提供了两种参数传递的方式:值传递和引用传递
值传递
在方法调用中,实参会把它的值传递给形参,形参只是用实参的值初始化一个临时的存储单元。因此形参与实参虽然有着响应的值,但是却有着不同的存储单元,因此对形参的改变不会影响实参的值。
引用传递
在方法调用中,传递的是是对象(也可以看作是对象的地址),这时候形参和实参的对象指的是同一块存储单元,因此对形参的修改就会影响实参的值。
在Java语言中,原始数据类型在传递参数时都是按值传递,而包装类在传递参数时是按引用传递的。
public class Test {
public static void main(String[] args) {
int i = 1;
StringBuffer stringBuffer = new StringBuffer("Hello");
testPassParameter(stringBuffer,i);
System.out.println(stringBuffer);
System.out.println(i);
}
private static void testPassParameter(StringBuffer s, int n) {
s.append(" World");
n = 8;
}
}
传递过程:
假设1和"Hello"的存储地址如图所示分别是0X12345678和0xFFFFFF12。在调用testPassParameter时,由于i为基本类型,因此参数是按值传递的,此时会创建一个i的副本,该副本与i具有相同的值,把这个副本作为参数赋值为n,作为传递的参数。
而StringBuffer是一个类,因此是按照引用传递,传递的是它的引用也就是"Hello"的地址
在testPassParameter内部修改的是n的值,这个值与i是没有关系的,但是在修改s时修改的是s这个地址指向的字符串,由于形参s和实参stringBuffer指向的是同一块存储空间,因此修改s后,s指向的字符串也被修改了。
Java中处理8种基本的数据类型用的是值传递,其他所有类型都用的是引用传递
public class Test {
public static void main(String[] args) {
Integer a = 1;
Integer b = a;
b++;
System.out.println(a);
System.out.println(b);
StringBuffer s1 = new StringBuffer("Hello");
StringBuffer s2 = new StringBuffer("Hello");
changeStringBuffer(s1,s2);
System.out.println(s1);
System.out.println(s2);
}
private static void changeStringBuffer(StringBuffer ss1, StringBuffer ss2) {
ss1.append(" World");
ss2 = ss1;
}
}
对于前两个输出一个"1"和"2",有人可能会认为Integer是按值传递。但其实是引用传递,只是由于Integer是不可变类,因此没有提供改变它值的方法,在上例中,在执行完b++后由于Integer是不可变类,因此没有提供改变它值的方法,在上例中,在执行完b++后,由于Integer是不可变类,因此此时会创建一个新值为2的Integer赋值给b,此时b与a已经没有任何关系了。
为了便于理解假设s1和s2指向的字符串地址分别为0X12345678和OXFFFFFF12,并且在参数传递是是按引用传递也就是s1和s2的地址。在调用方法ss1.append(“World”)时,会修改ss1所指向的字符串的值,因此会修改s1的值,得到输出结果为Hello World。但是在执行ss2 = ss1时,只会修改ss2的值而对s2的值毫无影响,因此s2在调用前后保持不变。