Java中的值传递与引用传递详解

方法调用是编程语言中非常重要的一个特性,在方法调用时,通常需要传递一些参数来完成特定的功能。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指向字符串的地址分别为0X123456780XFFFFFF12,那么在调用函数changeStringBuffer时,传递s1与s2的引用就可以理解为传递了两个地址0X123456780XFFFFFF12,而且这两个地址是按值传递的(即传递了两个值,ss1为0X12345678,ss2为0XFFFFFF12),在调用方法ss1.append(" World")时,会修改ss1所指向的字符串的值,因此会修改调用者的s1的值,得到的输出结果为“Hello World”。但是在执行ss2=ss1时,也就是把ss2指向了ss1指向的地址,只会修改ss2的值而对s2毫无影响,因此s2的值在调用前后保持不变。

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值