方法调用是编程语言中非常重要的一个特性,在方法调用时,通常需要传递一些参数来完成特定的功能。Java语言提供了两种参数传递的方式:值传递和引用传递。
(1)值传递
在方法调用中,实参会把它的值传递给形参,形参只是实参的值初始化一个临时的存储单元,因此形参与实参虽然有着相同的值,但是却有着不同的存储单元,因此对形参的改变不会影响实参的值。
(2)引用传递
在方法调用中,传递的是对象(也可以看作是对象的地址),这时形参与实参的对象指向同一块存储单元,因此对形参的改变就会影响实参的值。
在Java语言中,原始数据类型在传递参数时都是按值传递,而包装类型在传递参数时时按引用传递的。
下面通过一个例子来介绍按值传递和按引用传递的区别:
package com.js;
public class Test {
public static void testPassParameter(StringBuffer ss1,int n){
ss1.append(" World");
n=8;
}
public static void main(String[] args){
int i=1;
StringBuffer s1 = new StringBuffer("Hello");
testPassParameter(s1, i);
System.out.println(s1);
System.out.println(i);
}
}
运行结果:
Hello World
1
按引用传递其实与传递指针类似,是把对象的地址作为参数的,如下图所示:
为了便于理解,假设1和“Hello”存储的地址分别为0X12345678和0XFFFFFF12。在调用方法testPassParameter时,由于i为基本类型,因此参数是按值传递的,此时会创建一个i的副本,该副本与i有相同的值,把这个副本作为参数赋值给n,作为传递的参数。而StringBuffer由于是一个类,因此按引用传递,传递的是它的引用(传递的是存储“Hello”的地址),如上图所示,在testPassParameter内部修改的是n的值,这个值与i是没有关系的。但是在修改ss1时,修改的是ss1这个地址指向的字符串,由于形参ss1与实参s1指向的是同一块存储空间,因此修改ss1后,s1指向的字符串也被修改了。
Java中处理8种基本的数据类型用的是值传递,其他所有类型都是引用传递,由于这8种数据类型的包装类型都是不可变量,因此增加了对“按引用传递”的理解难度。下面请看一个例子:
package com.js;
public class Test {
public static void changeStringBuffer(StringBuffer ss1,StringBuffer ss2){
ss1.append(" World");
ss2=ss1;
}
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);
}
}
运行结果:
1
2
Hello World
Hello
对于上述的前两个输出“1”和“2”,有人会认为,Integer是按值传递的而不是按引用传递的。其实这是一个理解上的误区,上述代码还是按引用传递的,只是由于Integer是不可变类,因此没有提供改变它值的方法,在上例中,执行完语句b++后,由于Integer是不可变类,因此此时会创建一个新值为2的Integer赋值给b,此时b与a其实已经没有任何关系了。对于程序的后两个输出,可以加深对“按引用传递”的理解。如下图所示:
首先必须理解“引用 也是按值传递的”这一要点。为了便于理解,假设s1和s2指向字符串的地址分别为0X12345678和0XFFFFFF12,那么在调用函数changeStringBuffer时,传递s1与s2的引用就可以理解为传递了两个地址0X12345678和0XFFFFFF12,而且这两个地址是按值传递的(即传递了两个值,ss1为0X12345678,ss2为0XFFFFFF12),在调用方法ss1.append(" World")时,会修改ss1所指向的字符串的值,因此会修改调用者的s1的值,得到的输出结果为“Hello World”。但是在执行ss2=ss1时,也就是把ss2指向了ss1指向的地址,只会修改ss2的值而对s2毫无影响,因此s2的值在调用前后保持不变。