前段时间在群里看到类似这样一个问题,下面的代码会输出什么呢?
当时看到这题,瞬间勾起了我的回忆。遥想当年,也曾经碰到过类似的问题,当时研究了好久才搞明白,这里再记录一下这个问题的思路。
先来说一下答案:输出:hello;
解决这类问题首先要搞明白Java到底是引用传递还是值传递。
首先来解释一下什么是引用传递,什么是值传递。
那在Java中到底是引用传递还是值传递呢?其实这个问题也一直是争论不断,而且官方也没给个确切答案。但是就我个人理解,Java是值转递。
我们先来看一个简单的例子:
a的值:1
在test()方法中定义了一个基本类型的变量a,然后调用change()方法试图改变这个变量,最后输出的还是原来的值。
首先我们要清楚,一个方法中的局部变量是存在栈中的,如果是基本类型的变量则直接存的是这个变量的值,如果是引用类型的变量则存的是值的地址,指向堆中具体的值。
上面的例子中,调用change()方法传递的a,其实是a变量的拷贝,不是真正的a,在change()方法中改变的是拷贝,对真正的a是没有影响的。
这么一看,Java确实是值传递,但是我们再看下面这个例子,你就会纠结了
// 输出 年龄:19
看,对象里的属性被改变了,不是值传递吗,应该不会改变啊,这时候就有人总结了,当传递值是基本类型时是值传递、当传的是引用类型时是引用传递。真的是这样吗?
分析这个问题,我们需要知道变量在jvm中是怎么存储的。
首先看基本类型,这个很简单,变量在栈中直接存的是值,传到change()方法的是这个变量的拷贝,因此对拷贝的变量修改不会影响原变量的值。
Java是值传递。当传的是基本类型时,传的是值的拷贝,对拷贝变量的修改不影响原变量;当传的是引用类型时,传的是引用地址的拷贝,但是拷贝的地址和真实地址指向的都是同一个真实数据,因此可以修改原变量中的值;当传的是String类型时,虽然拷贝的也是引用地址,指向的是同一个数据,但是String的值不能被修改,因此无法修改原变量中的值。