Java开发那些事Java 杂谈Java面试

Java 中的值传递

2017-11-17  本文已影响38人  Xiao_Mai

其实 Java 中是只存在值传递的,不存在引用传递。因为我们大多数人是从 C 语言入门,而 C 语言中是存在引用传递的,所以很容易在 Java 中混淆。

数据类型分类

Java 中的数据类型分为两大类:基本类型对象类型,相对应的变量也分为两类:基本类型引用类型

// int 基本类型的数据类型,num 基本类型的变量,变量的值 10 直接保存在变量 num 中。
int num = 10;
// String 对象类型的数据类型,str 引用类型的变量,str 中保存的是 "hello" 在内存中的首地址。
String str = "hello";

下面这张图代表了变量中保存的值,以及实际对象在内存中的情况。


image

当修改变量的值时:

num = 20;
str = "java";
image

通过上图可以看出:

参数传递

这里首先要记住一点:参数传递就是赋值操作,也就是说,我们调用一个带参的函数时,形参和实参是两个变量,在内存中也是开辟了两个空间,只是把实参的值赋值给了形参。这是理解 Java 中只有值传递的关键。

下面就通过几个例子证明这一点。

例1.基本类型的变量传参

基本类型的变量传参

输出结果:

99

这个很好理解,其实通过编译器也可以发现,foo() 方法中的 num 颜色为灰色并且带有波浪线,把鼠标放到变量上还会看到提示:“Parameter can be converted to a local variable”。意思就是这个变量可以转化为一个本地变量,也就是在方法内部声明。根据“参数传递就是赋值操作”,来梳理一下整个过程。当调用 foo() 方法时,将实参( num) 的值赋值给了形参(num1 ),这时内存中存在两个变量 num(实参)、num1(形参),因为基本类型的变量赋值是直接覆盖操作,所以我们对形参(num1)的操作只是改不了形参的值,而不会影响实参(num) 。

例2.普通引用类型的变量传参

说是普通引用类型,是指类内部提供了改变自身的方法。

普通引用类型的变量传参

这次可以发现编译器没有提示异常信息,这至少可以证明,在 foo() 方法中对形参 myObject1 进行操作之后,形参 myObject1 仍然被引用,那是不是可以证明形参 myObject1 和实参 myObject 是同一个对象呢?我们先看看结果:

100

结果证明,foo 方法中的实参 myObject 和形参 myObject1 确实是指向同一个对象。再根据“参数传递就是赋值操作”这句话,来梳理一下。当调用 foo() 方法时,将实参(myObject) 的值(引用对象的内存地址)赋值给形参 (myObject1),这时内存中存在两个变量 myObject(实参)、myObject1(形参),但是它们都指向同一个内存地址,所以我们对形参(myObject1)的操作会影响实参(myObject) 。

例3.特殊引用类型的变量传参

上面提到了普通引用类型,当然就存在特殊引用类型了,它是指自身保存的值不可修改。如:String 和 Integer、Double、Boolean 等基本类型的包装类,它们都是 Immutable 类型的(它们的值都是 final 修饰的),所以每次对它们进行赋值操作,都是创建一个新的对象。

特殊引用类型的变量传参

我们发现 foo() 方法中的 value 被编译器提示可修改为本地变量。这就和例 1 中一样在 foo 方法中对形参的修改不会影响到实参,输出结果:

hello

这次根据“参数传递就是赋值操作”以及特殊引用类型的特点来梳理一下。当调用 foo() 方法时,将实参(value) 的值(引用对象的内存地址)赋值给形参 (value1),这时内存中存在两个变量 value(实参)、value1(形参),而且它们都指向同一个内存地址,但是当我们对形参(value1)进行赋值操作时,因为它是一个自身不可修改的特殊引用类型,所以"hello"对象并没有修改,而是在内存中又创建了一个值为"java”的新对象,并把“java”的地址赋值给形参,所以只是形参变了,而实参还是指向“hello”对象。

例4. 对普通引用类型使用赋值运算符

这次把 foo() 方法进行了修改,在方法内对形参进行重新赋值操作。

对普通引用类型使用赋值运算符

这时编译器又提示了形参 myObject 可以修改为本地变量。再根据“参数传递就是赋值操作”这句话,来梳理一下。当调用 foo() 方法时,将实参(myObject) 的值(引用对象的内存地址)赋值给形参 (myObject1),这时内存中存在两个变量 myObject(实参)、myObject1(形参),而且它们都指向同一个内存地址,但是在方法内部又对形参进行了一次赋值操作,这时形参指向了一个新的对象,而实参仍然指向原来的对象,这样形参和实参之间没有任何关联了。所以我们对形参(myObject1)的操作不会影响实参(myObject) 。

其实 foo() 方法等同于:

private static void foo() {
    MyObject myObject = new MyObject();
    myObject.num = 100;
}

同样可以看出 foo() 方法中的形参和实参完全没有关系了。

总结

Java 中参数传递其实就是赋值操作。
Java 中只存在值传递。

上一篇下一篇

猜你喜欢

热点阅读