这个问题,一直很困惑。不过今天看到一篇文章,对Java到底是传引用还是传值的问题有了一些总结。
文章中的原话是这样说的“Java确实使用对象的引用来做计算的,所有的对象变量都是引用。但是,Java在向方法传递参数时传的不是引用,是值。”。现在说一下我的理解。
badSwap() 函数:
public void badSwap(int var1, int var2)
{
int temp = var1;
var1 = var2;
var2 = temp;
}
当badSwap方法返回时,被当作参数传入的变量仍然保持了原来的值不变。如果我们把传入的int型变量改为Object型也是一样的。
代码的相关实例:
package cn.com.webjg.guilin;
import java.awt.Point;
public class test {
public static void main(String[] args) {
Point prot1 = new Point(0,0);
Point prot2 = new Point(0,0);
System.out.println("=======对象初始=========");
System.out.println("X:" + prot1.x + " Y:" + prot1.y);
System.out.println("X:" + prot2.x + " Y:" + prot2.y);
System.out.println(" ");
trickey(prot1, prot2);
System.out.println("=======调用trickey方法后的外部输出,副本引用交换对象后 的结果=========");
System.out.println("X:" + prot1.x + " Y:" + prot1.y);
System.out.println("X:" + prot2.x + " Y:" + prot2.y);
System.out.println("=======在trickey函数外部执行原始引用交换对象的结果=========");
Point temp2 = prot1;
prot1 = prot2;
prot2 = temp2;
System.out.println("X:" + prot1.x + " Y:" + prot1.y);
System.out.println("X:" + prot2.x + " Y:" + prot2.y);
}
public static void trickey(Point arg1, Point arg2){
arg1.x = 100;
arg1.y = 100;
System.out.println("=======trickey函数中的对象修改对象值的结果=========");
System.out.println("X:" + arg1.x + " Y:" + arg1.y);
System.out.println("X:" + arg2.x + " Y:" + arg2.y);
Point temp = arg1;
arg1 = arg2;
arg2 = temp;
System.out.println("=======trickey函数中副本引用交换后的结果=========");
System.out.println("c-X:" + arg1.x + " c-Y:" + arg1.y);
System.out.println("c-X:" + arg2.x + " c-Y:" + arg2.y);
System.out.println(" ");
}
}
输出结果:
=======对象初始=========
X:0 Y:0
X:0 Y:0
=======trickey函数中的对象修改对象值的结果=========
X:100 Y:100
X:0 Y:0
=======trickey函数中副本引用交换后的结果=========
c-X:0 c-Y:0
c-X:100 c-Y:100
=======调用trickey方法后的外部输出,副本引用交换对象后的结果=========
X:100 Y:100
X:0 Y:0
=======在trickey函数外部执行原始引用交换对象的结果=========
X:0 Y:0
X:100 Y:100
从结果可以发下,trickey函数成功的修改了prot1的值,但是prot1和prot2的置换失败了。
这就意味着:传向函数的引用实际上是原始引用的副本。下面的图一展现了当Java传递对象给函数之后,两个引用指向了同一对象。(在此,原来引用和副本引用都指向了原来的对象,这里就是为什么trickey函数可以修改prot1的值。换句话说就是这个对象现在有2个引用。)
图1
因为方法中对象的引用是“副本”,所以对象交换就没起作用。交换动作只对trickey方法中的引用副本起作用了,不影响方法外的引用。所以,方法被调用后,改变不了方法外的对象的引用。如果要对方法外的对象引用做交换,我们应该交换原始的引用,而不是它的副本。(查看trickey函数中副本引用交换后的结果的输出,函数中2个参数确实置换了,可是函数中置换的是2个对象的副本引用,不会影响到原始引用。在main()函数当中,prot1和prot2是对象的原始引用。)