Java到底是传引用还是传值?

问题: 如果Java是用引用来传递的话,为什么交换函数(swap)不起作用呢?

回答: 你的问题引出了Java新手的常犯的错误。事实上,一些老手也很难搞清楚这些概念。

Java确实使用对象的引用来做计算的,所有的对象变量都是引用。但是,Java在向方法传递参数时传的不是引用,是值。

以 badSwap() 函数为例:

 public void badSwap(int var1, int var2)
 {
     int temp = var1;
     var1 = var2;
     var2 = temp;
 }

当badSwap方法返回时,被当作参数传入的变量仍然保持了原来的值不变。如果我们把传入的int型变量改为Object型也是一样的,因为Java通过传值来传递引用的。现在,我们来看下是哪个地方搞的鬼:

 public void tricky(Point arg1, Point arg2)
 {
     arg1.x = 100;
     arg1.y = 100;
     Point temp = arg1;
     arg1 = arg2;
     arg2 = temp;
 }

 public static void main(String [] args)
 {
     Point pnt1 = new Point(0,0);
     Point pnt2 = new Point(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

X: 100 Y: 100
X: 0 Y: 0
———————————————————-
即使是通过值传递,tricky函数依然成功地改变了pnt1的值。但是pnt1和pnt2的置换失败了。这正是最令人困惑的地方。在main()函数当中,pnt1和pnt2仅仅是对象的引用。当你向tricky()函数传递pnt1和pnt2参数时,Java仅仅向传递任何其他参数一样,通过传值来传递引用。这就意味着:传向函数的引用实际上是原始引用的副本。下面的图一展现了当Java传递对象给函数之后,两个引用指向了同一对象

图一: 当被传递给函数之后,一个对象至少存在两个引用

Java复制并传递了“引用”的值,而不是对象。因此,方法中对对象的计算是会起作用的,因为引用指向了原来的对象。但是因为方法中对象的引用是“副本”,所以对象交换就没起作用。如图2所示,交换动作只对方法中的引用副本起作用了,不影响方法外的引用。所以不好意思,方法被调用后,改变不了方法外的对象的引用。如果要对方法外的对象引用做交换,我们应该交换原始的引用,而不是它的副本。

图二: 只有传入函数的引用交换了,原始引用则没有

 

英文出处:JavaWorld  感谢新浪微博:@for-each @TizzyBec 翻译本文。

译文链接:http://www.importnew.com/3559.html

【如需转载,请在正文中标注并保留原文链接、译文链接和译者等信息,谢谢合作!】

关于作者: ImportNew读者

标注为 "ImportNew读者" 发布的文章,均为热心读者的投稿。感谢支持!

查看ImportNew读者的更多文章 >>



可能感兴趣的文章

发表评论

Comment form

(*) 表示必填项

7 条评论

  1. 余松 说道:

    “事实上,一些老手也很难搞清楚这些概念。”确实是这样
    平安银行的一个HR面试我的时候,我说方法传递参数的时候是传值,他偏说我是错的。

    Thumb up 0 Thumb down 1

  2. anonym 说道:

    小基础,大智慧。深感自己基础薄弱。

    Thumb up 0 Thumb down 0

  3. 余松 说道:

    这一点和ECMAScript 中 所有函数的参数都是按值传递原理是一样的。

    Thumb up 0 Thumb down 0

  4. 史以兵 说道:

    大家可以去看看C++,传值、传引用、传指针的区别是很明显的。Java中的参数传递其实就是C++中的传引用(基本类型可能被优化成传值了)。另外就是,Java中没有指针,所以没法传指针,swap也就无法工作了。

    Thumb up 0 Thumb down 0

    • anonym 说道:

      明显理解错了。java里的引用其实就是受限的指针,和C++的“引用”没啥关系。

      Thumb up 0 Thumb down 0

  5. netatomy 说道:

    在Java中,变量之间传递的是变量所存储的值,而不是传递变量的引用。对于基本类型的变量,存储的就是基本类型的值本身,对于对象的变量,存储的值是对那个对象的引用。

    Thumb up 1 Thumb down 0

  6. lin 说道:

    受教了,谢谢

    Thumb up 1 Thumb down 0

跳到底部
返回顶部