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毫无影响,因为此时,ss2 和s2 已经没在关系了,不同的地址,ss2其实就是ss1了,ss2和s2完全没有关系了,因此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毫无影响,此时ss2和s2是两个地址,ss2和ss1是一个地址了,所以和s2无关了。因此s2的值在调用前后保持不变,仍然是Hello。



  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值