问题: 若是Java是用引用来传递的话,为何交换函数(swap)不起做用呢? java
回答: 你的问题引出了Java新手的常犯的错误。事实上,一些老手也很难搞清楚这些概念。 函数
Java确实使用对象的引用来作计算的,全部的对象变量都是引用。可是,Java在向方法传递参数时传的不是引用,是值。 spa
以 badSwap() 函数为例: 对象
1
2
3
4
5
6
|
publicvoidbadSwap(intvar1,intvar2)
{
inttemp = var1;
var1 = var2;
var2 = temp;
}
|
当badSwap方法返回时,被看成参数传入的变量仍然保持了原来的值不变。若是咱们把传入的int型变量改成Object型也是同样的,由于Java经过传值来传递引用的。如今,咱们来看下是哪一个地方搞的鬼: ci
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
publicvoidtricky(Point arg1, Point arg2)
{
arg1.x =100;
arg1.y =100;
Point temp = arg1;
arg1 = arg2;
arg2 = temp;
}
publicstaticvoidmain(String [] args)
{
Point pnt1 =newPoint(0,0);
Point pnt2 =newPoint(0,0);
System.out.println("X: "+ pnt1.x +" Y: "+pnt1.y);
System.out.println("X: "+ pnt2.x +" Y: "+pnt2.y);
System.out.println(" ");
tricky(pnt1,pnt2);
System.out.println("X: "+ pnt1.x +" Y:"+ pnt1.y);
System.out.println("X: "+ pnt2.x +" Y: "+pnt2.y);
}
|
执行这个函数,将获得如下输出:
———————————————————-
X: 0 Y: 0
X: 0 Y: 0 get
X: 100 Y: 100
X: 0 Y: 0
———————————————————-
即便是经过值传递,tricky函数依然成功地改变了pnt1的值。可是pnt1和pnt2的置换失败了。这正是最使人困惑的地方。在main()函数当中,pnt1和pnt2仅仅是对象的引用。当你向tricky()函数传递pnt1和pnt2参数时,Java仅仅向传递任何其余参数同样,经过传值来传递引用。这就意味着:传向函数的引用其实是原始引用的副本。下面的图一展示了当Java传递对象给函数以后,两个引用指向了同一对象 it
图一: 当被传递给函数以后,一个对象至少存在两个引用 table
Java复制并传递了“引用”的值,而不是对象。所以,方法中对对象的计算是会起做用的,由于引用指向了原来的对象。可是由于方法中对象的引用是“副本”,因此对象交换就没起做用。如图2所示,交换动做只对方法中的引用副本起做用了,不影响方法外的引用。因此很差意思,方法被调用后,改变不了方法外的对象的引用。若是要对方法外的对象引用作交换,咱们应该交换原始的引用,而不是它的副本。 import
图二: 只有传入函数的引用交换了,原始引用则没有 stream
我的理解:java参数的传递传的是引用,可是是经过传值的方式来传递引用,也就是说传过来的参数不是原始对象,而是建立了一个新的对象,只不过这个新的对象与原始对象都是同一个引用。因此副本对象属性的改变会反应到原始对象,可是副本对象经过从新赋值改变的只是自己的引用。因此对象的交换不起做用,由于改变的只是副本的引用,没有改变原始对象的引用。