Object类是所有类的始祖,在java中每个类都是由它扩展而来的。但是并不需要加上extends关键字,如果没有明确地指出超类,Object就被认为是这个类的超类。由于在Java中,每个类都是由Object类扩展而来的,所以,熟悉这个类提供的所有服务十分重要。在本文结尾将会附上Object类源码。
equals方法用于检测一个对象是否等于另一个对象。在Object类中,这个方法将判断两个对象是否具有相同的引用。如果两个对象具有相同的引用,它们一定是相等的。equals常用与if判断中。
上述代码的功能是比较"123456"字符串是否与user的密码相等,如果不相等则输出"密码错误"。
==: 对于基本类型 ,比较两者数值是否相等;对于引用类型,比较两个对象的内存地址是否相等。(同一个new出来的对象,或存放进常量池的对象使用==是相等的) equals(): 比较两个对象存放的内容是否相等。
因此在对象比较的时候最好使用equals方法来进行比较。
在大部分情况下,这种判断并没有什么意义,因此需要重写equals方法。比如使用两者的唯一标示id来作为判断是否相等的依据。同时重写的时候需要遵守Java语言规范要求的equals的几个特性: 1)自反性:对于任何非空引用x,x.equals(x)应该返回true。 2)对称性:对于任何引用x和y,当且仅当y.equals(x)返回true,x.equals(y)也应该返回true。 3)传递性:对于任何引用x、y和z,如果x.equals(y)返回true,y.equals(x)返回true,x.equals(z)也应该是true。 4)一致性:如果x和y引用的对象没有发生变化,反复调用x.equals(y)应该返回同样的结果。 5)对于任意非空引用x,x.equals(null)应该返回false。
重写示例:
public class test { ... @Override public boolean equals(Object otherObject) { //快速比较,相同引用返回true if(this==otherObject) { return true; } //如果为null一定要返回false if (otherObject == null) { return false; } //如果class没有匹配,即两者不是一个类,返回false if (getClass() != otherObject.getClass()) { return false; } //非空 Employee other = (Employee) otherObject; //测试两者字段是否有相同值 return name.equals(other.name) && salary == other.salary && hireDay.equals(other.hireDay); } }1)显示参数命名为otherObject,稍后需要将它转换成另一个叫做other的变量;
2)检测this与otherObject是否引用同一个对象:
if(this == otherObject) return true;
3)检测otherObject是否为null,如果为null,返回false:
if(otherObject==null) return false;
4)比较this与otherObjcet是否属于同意各类。如果equals的语义在每个子类中有所改变,就使用getClass检测:
if(getClass() != otherObject.getClass()) return false;
如果所有的子类都拥有统一的语义,就使用instanceof检测:
if(!(otherObject instanceof ClassName)) return false;
5)将otherObject转换为相应的类类型变量:
ClassName other = (ClassName) otherObject;
6)现在开始对所有需要比较的域进行比较了。使用==比较基本类型域,使用equals比较对象域。如果所有的域都匹配,就返回 true;否则返回false。
return field1 == other.field1 && Objects.equals(field2,other.field2) && ...;
其中第四步是为了保证equals的对称性原则。如果在子类中重新定义equals,就要在其中包含调用super.equals(other)。
散列码(hash code)是由对象导出的一个整型值。散列码是没有规律的。如果x和y是两个不同的对象,x.hashCode()与y.hashCode()基本上不会相同。由于hashCode方法定义在Object类中,因此每个对象都有一个默认的散列码,其值为对象的存储地址。如果重写了equals方法,那么就需要重写hashCode方法。
equals与hashCode的定义必须一致:如果x.equals(y)返回true,那么x.hashCode()就必须与y.hashCode()具有相同的值。
重写示例:
public class test { ... @Override public int hashCode() { return Objects.hash(name, salary, hireDay); } }Objects.hash()方法会对各个参数调用Objects.hashCode,并组合这些散列值。
toString方法是一个十分重要的方法,它用于返回表示对象值的字符串。只要对象与一个字符串通过操作符“+”连接起来,Java编译就会自动地调用toString方法,以便获得这个对象的字符串描述。
在Object类中,toString会打印对象所属的类名和散列码。例如,调用System.out.println(System.out)将输出java.io.PrintStream@2f6684,观感并不是很好。
因此,强烈建议为自定义的每一个类增加toString方法。这样做不仅自己受益,而且所有使用这个类的程序员也会从这个日志记录支持中受益匪浅。
重写示例:
public class test { ... @Override public String toString(){ return getClass().getName() + "[name=" + name + ",salary=" + salary + "]"; } }