随笔-16  评论-50  文章-2  trackbacks-0

在程序开发的过程,要交换两个变量的内容,是一种比较常见的事情。在排序算法中,就有一种就叫做“交换排序法”。在所有的排序算法,交换要排序的集合中的两个元素,几乎是必须的过程。在Java中交换两个元素的内容,如果你是程序员新手,你可能碰到意想不到的问题。

首先,来看看交换下面的程序。为了交换两个整数变量的内容,写了下面这样的一个方法实现:

public void swap(int i, int j) {
    int t = i;
    i = j;
    j = t;
}

调用上面的方法,有问题吗?

int x = 100;
int y = 1;
swap(x, y);

在程序执行此段代码后,你会发现,x还是100, y还是1。为什么呢?因为Java对方法参数的传递,是使用值调用(call by value)的。想想,如果我这样调用swap函数呢,swap(3, 4),这是莫名其妙的,有谁会要交换3和4这两个常数呢。

那没办法交换两个整数变量了吗?可以。把swap函数体替换掉swap函数的调用就可以了。如:

int x = 100;
int y = 1;
int t = x;
x = y;
y = t;

Java中其他的原生类型(primitive type)的情况,和int的一样。

 

继续讨论函数调用吧。函数调用时,参数传递的方式主要有两种:

  • 值传递
  • 引用传递

Java使用的是值传递。值传递是把变量的值、常数或常量传递给参数。而引用传递,是把变量的所在内存中的地址传递给参数,参数通过地址找到变量的值。很明显,引用传递不能把常数传递给参数。值传递和引用传递还有一个很大的不同:对于像int这样的小类型变量来说,值传递没副作用,而引用传递有。也就是说,在函数调用的执行过程中,不能改变传递给参数的变量的值。

但对于普通类类型参数的传递方式的理解和原生类型有点不同。对于方法method(Object o)的调用method(x), 不是把对象x复制一份传递给参数o,而是把对象x的在内存中的首地址,也就是把对象x的引用拷贝给参数o。这样就能这样实现对象的交换函数了吗?

public void swap(Object o, Object p) {
    Object t = o;
    o = p;
    p = t;
}

答案是:No。因为像下面这样的调用:

Object x = X;
Object y = Y;
swap(x, y);

在执行完上的代码后,x指向的还是原来的X对象,y指向的还是那个Y对象。就像歌唱的那样:星星还是那个星星,月亮还是那个月亮。

 

难道就不能通过方法调用实现交换这个功能吗?可以。有两种办法:

  • 可以交换数组等集合里的元素
  • 用反射实现

public void swap(Object[] a, int i, int j) {
    Object t = a[i] ;
    a[i] = a[j];
    a[j] = a[i];
}

用反射实现swap函数,有点杀鸡用大炮的感觉。性能不怎么样,还容易出错。这个留着做家庭作业吧。

posted on 2008-01-03 10:42 Jeff Lau 阅读(1135) 评论(1)  编辑  收藏 所属分类: Jeff On Java 2008

评论:
# re: 交换[未登录] 2008-04-02 19:58 | Tom
最后的swap(Object[] a, int i, int j)中第三个语句写错了吧?
应该是a[j] = t;  回复  更多评论
  

只有注册用户登录后才能发表评论。


网站导航: