今天,我在一本面试书上看到了关于java的一个参数传递的问题:javascript
我毫无疑问的回答:“引用传递!”,而且还以为本身对java的这一特性非常熟悉!html
结果发现,我错了!java
答案是:程序员
值传递!Java中只有按值传递,没有按引用传递!web
回家后我就火烧眉毛地查询了这个问题,以为本身对java这么基础的问题都搞错实在太丢人!面试
综合网上的描述,我大概了解了是怎么回事,如今整理以下,若有不对之处望大神提出!编程
先来看一个做为程序员都熟悉的值传递的例子:app
答案显而易见,调用函数changeValue()先后num的值都没有改变。编程语言
由此作一个引子,我用图表描绘一个值传递的过程:函数
num做为参数传递给changeValue()方法时,是将内存空间中num所指向的那个存储单元中存放的值,即"5",传送给了changeValue()方法中的x变量,而这个x变量也在内存空间中分配了一个存储单元,这个时候,就把num的值5传送给了这个存储单元中。此后,在changeValue()方法中对x的一切操做都是针对x所指向的这个存储单元,与num所指向的那个存储单元没有关系了!
天然,在函数调用以后,num所指向的存储单元的值仍是没有发生变化,这就是所谓的“值传递”!值传递的精髓是:传递的是存储单元中的内容,而非地址或者引用!
接下来,就来看java中的对象参数是怎么传递的:
一样,先给出一段代码:
答案应该你们都心知肚明:
第一次显示:“Jack”
第二次显示:“Rose”
方法用了一个对象参数,该对象内部的内容就能够改变,我以前一直认为应该是该对象复制了一个引用副本给调用函数的参数,使得该方法能够对这个对象进行操做,实际上是错了!
为何这里是“值传递”,而不是“引用传递”?
我仍是用图表描绘比较能解释清楚:
主函数中new 了一个对象Person,实际分配了两个对象:新建立的Person类的实体对象,和指向该对象的引用变量person。
【注意:在java中,新建立的实体对象在堆内存中开辟空间,而引用变量在栈内存中开辟空间】
正如如上图所示,左侧是堆空间,用来分配内存给新建立的实体对象,红色框是新建的Person类的实体对象,000012是该实体对象的起始地址;而右侧是栈空间,用来给引用变量和一些临时变量分配内存,新实体对象的引用person就在其中,能够看到它的存储单元的内容是000012,记录的正是新建Person类实体对象的起始地址,也就是说它指向该实体对象。
这时候,好戏上台了:
调用了changeName()方法,person做为对象参数传入该方法,可是你们特别注意,它传入的是什么!!!person引用变量将本身的存储单元的内容传给了changeName()方法的p变量!也就是将实体对象的地址传给了p变量,今后,在changeName()方法中对p的一切操做都是针对p所指向的这个存储单元,与person引用变量所指向的那个存储单元再没有关系了!
回顾一下上面的一个值传递的例子,值传递,就是将存储单元中的内容传给调用函数中的那个参数,这里是否是殊途同归,是所谓“值传递”,而非“引用传递”!!!
那为何对象内部可以发生变化呢?
那是由于:p所指向的那个存储单元中的内容是实体对象的地址,使得p也指向了该实体对象,因此才能改变对象内部的属性!
这也是咱们大多数人会误觉得是“引用传递”的终极缘由!!!