在这篇文章里我将介绍一些Java对于对象的基本处理模式的问题。
首先看一个有关对象相等性(Object equivalence)的例子:
先看下面的代码:
public class EqualTest1{
public static void main(String[] args){
Integer n1 = new Integer(20);
Integer n2 = new Integer(20);
System.out.println(n1 = = n2);
System.out.println(n1 != n2);
}
}
程序的目的是输出括号中的比较结果(boolean值),初次接触Java的人很轻易的认为输出结果为先true而后false。
但实际上结果是先false而后true,因为虽然两个Integer对象的值是相同的,其reference却不同。(注:有关reference的含义在我的上一篇学习笔记上有过介绍,这里不再赘述。)
为了解释上面这个问题,我们应该了解Java对于对象的基本处理模式:
当你操作某个对象时,你所操作的其实是它的reference,比如A = B这个式子,就会将A和B都指向原来B所指向的对象,假如你改变了A的内容,那么同时也就更改了B的内容!因为A和B内含同一个object reference。
原先A中所储存的reference,在赋值的过程中被覆写了,实际上就是遗失掉了,因为垃圾回收器(garbage collector)会在适当时机清理该reference原本指向的那个对象。
那么如何知道对象的内容是否相等呢?这里就要用到equals(),请看下面的代码:
public class EqualTest2{
public static void main(String[] args){
Integer n1 = new Integer(20);
Integer n2 = new Integer(20);
System.out.println(n1.equals(n2));
}
}
这样输出的就是我门所期望的true了。然后,事情并不会如此简单,假如建立自有的class,那么事情会怎么样呢?请看相面的代码:
class Value{
int i;
}
public class EqualTest3{
public static void main(String[] args){
Value v1 = new Value();
Value v2 = new Value();
v1.i = v2.i = 20;
System.out.println(v1.equals(v2));
}
}
结果又输出了false,这是为什么呢??
其实,equals()的缺省行为是拿reference来比较,所以除非在你的class中覆写(override)equals(),否则不会得到预期的结果,而Java标准程序库中的大多数class都覆写了equals(),所以他们都会比较对象的内容是否相同,这样一来,上面的问题就不难解决了。
有关覆写(override)技术将在今后深入的文章中介绍:)