在传递引用的时候其实是复制了一份引用传进去的.
A a=new A();
test(a)
相当于
(A b=a;
test(b)
)
如果传的是基本数据类型,修改这个值并不会影响作为参数传进来的那个变量,因为你修改的是方法的局部变量,是一个副本。
如果传的是一个对象的引用,也是一样的,也是一个副本,但是这个副本和作为参数传进来的那个引用指向的是内存中的同一个对象,所以你通过这个副本也可以操作那个对象。但是如果你修改这个引用本身,比如让他指向内存中的另外一个对象,原来作为参数传进来的那个引用不会受到影响。
我觉得弄明白这些就行了,说值传递或引用传递都无所谓,但是说值传递更适合一些,这个值可以是引用也可以是基本数据类型。
--------- 引用他人的话
所有的调用都是按值传递这是没错的,因为调用堆栈的原理限定了我们只能将各种值压入堆栈,而方法返回时,并不会将堆栈中的值再进行处理,而只是简单的调整栈顶指针将原先压入堆栈的值废弃掉。所以,一切对压栈而传递到方法体内的参数,方法内部所做的修改对外界都是无法看到的。那么鉴于这种情况,为了将函数内部对参数的修改可以带到函数外,各种语言做了不同的处理,C/C++中可以传递指针,而Java则默认传递对象的引用。如果楼主非要把方法调用时压入堆栈的地址称为值的话,其实也并非不可,只不过这种说法我觉得有点儿牵强了。
--------- 引用他人的话
同意两位的意见!谢谢
今天,我在一本面试书上看到了关于Java的一个参数传递的问题:
我毫无疑问的回答:“引用传递!”,并且还觉得自己对java的这一特性很是熟悉!
结果发现,我错了!
答案是:
值传递!Java中只有按值传递,没有按引用传递!
回家后我就迫不及待地查询了这个问题,觉得自己对java这么基础的问题都搞错实在太丢人!
综合网上的描述,我大概了解了是怎么回事,现在整理如下,如有不对之处望大神提出!
先来看一个作为程序员都熟悉的值传递的例子:
- ... ...
- //定义了一个改变参数值的函数
- public static void changeValue(int x) {
- x = x *2;
- }
- ... ...
- //调用该函数
- int num = 5;
- System.out.println(num);
- changeValue(num);
- System.out.println(num);
- ... ...
答案显而易见,调用函数changeValue()前后num的值都没有改变。
由此做一个引子,我用图表描绘一个值传递的过程:
num作为参数传递给changeValue()方法时,是将内存空间中num所指向的那个存储单元中存放的值,即"5",传送给了changeValue()方法中的x变量,而这个x变量也在内存空间中分配了一个存储单元,这个时候,就把num的值5传送给了这个存储单元中。此后,在changeValue()方法中对x的一切操作都是针对x所指向的这个存储单元,与num所指向的那个存储单元没有关系了!
自然,在函数调用之后,num所指向的存储单元的值还是没有发生变化,这就是所谓的“值传递”!值传递的精髓是:传递的是存储单元中的内容,而非地址或者引用!
接下来,就来看java中的对象参数是怎么传递的:
同样,先给出一段代码:
- ... ...
- class person {
- public static String name = "Jack";
- ... ...
- }
- ... ...
- //定义一个改变对象属性的方法
- public static void changeName(Person p) {
- p.name = "Rose";
- }
- ... ...
- public static void main(String[] args) {
- //定义一个Person对象,person是这个对象的引用
- Person person = new Person();
- //先显示这个对象的name属性
- System.out.println(person.name);
- //调用changeName(Person p)方法
- changeName(person);
- //再显示这个对象的name属性,看是否发生了变化
- System.out.println(person.name);
- }
答案应该大家都心知肚明:
第一次显示:“Jack”
第二次显示:“Rose”
方法用了一个对象参数,该对象内部的内容就可以改变,我之前一直认为应该是该对象复制了一个引用副本给调用函数的参数,使得该方法可以对这个对象进行操作,其实是错了!
为什么这里是“值传递”,而不是“引用传递”?
我还是用图表描绘比较能解释清楚:
主函数中new 了一个对象Person,实际分配了两个对象:新创建的Person类的实体对象,和指向该对象的引用变量person。
【注意:在java中,新创建的实体对象在堆内存中开辟空间,而引用变量在栈内存中开辟空间】
正如如上图所示,左侧是堆空间,用来分配内存给新创建的实体对象,红色框是新建的Person类的实体对象,000012是该实体对象的起始地址;而右侧是栈空间,用来给引用变量和一些临时变量分配内存,新实体对象的引用person就在其中,可以看到它的存储单元的内容是000012,记录的正是新建Person类实体对象的起始地址,也就是说它指向该实体对象。
这时候,好戏上台了:
调用了changeName()方法,person作为对象参数传入该方法,但是大家特别注意,它传入的是什么!!!person引用变量将自己的存储单元的内容传给了changeName()方法的p变量!也就是将实体对象的地址传给了p变量,从此,在changeName()方法中对p的一切操作都是针对p所指向的这个存储单元,与person引用变量所指向的那个存储单元再没有关系了!
回顾一下上面的一个值传递的例子,值传递,就是将存储单元中的内容传给调用函数中的那个参数,这里是不是异曲同工,是所谓“值传递”,而非“引用传递”!!!
那为什么对象内部能够发生变化呢?
那是因为:p所指向的那个存储单元中的内容是实体对象的地址,使得p也指向了该实体对象,所以才能改变对象内部的属性!
这也是我们大多数人会误以为是“引用传递”的终极原因!!!