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指向字符串的地址分别为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的值在调用前后保持不变。
 

猜你喜欢

转载自blog.csdn.net/lkp1603645756/article/details/83818959