类的方法前加final关键字,说明该方法不能被该类的子类重写。
如果类具有自己特有的“逻辑相等”概念(不同于对象等同的概念),而且超类还没有覆盖equals以实现期望的行为,这时候我们就需要覆盖equals方法。
equals方法实现了等价关系:自反性,对称性,传递性,一致性,非空性(x.equals(null)返回为false)
class Student{ private String name; private int age; private double height; @Override public boolean equals(Object obj) { //使用==操作符检查“参数是否为这个对象的引用” if(this==obj) return true; //使用instanceof操作符检查“参数是否为正确的类型” if(!(obj instanceof Student)) return false; //把参数转换成正确的类型。 Student student=(Student) obj; //对于该类中的每个“关键”阈,检查参数中的阈是否与该对象中对应的阈相匹配。 return this.name==student.name && this.age==student.age && this.height==student.height; } }
1、编写完成equals方法之后,应该问自己三个问题:它是否对称的、传递的、一致的。
2、不要将equals声明中的Object对象替换为其他的类型
public boolean equals(Student obj)
这样相当于重载了equals方法,而非是覆盖。
HashCode有一条约定如下:
如果两个对象根据equals(Object)方法比较是相等,那么调用这两个对象中任意一个对象的hashCode方法都必须产生同样的整数结果。
下面给出一种简单的解决办法:
private boolean flag=true; int boolTemp=flag?0:1;
private double height; long heightBits=Double.doubleToLongBits(height); int heightTemp=(int)(heightBits ^ (heightBits >>> 32));
private String name; int stringTemp=this.name.hashCode();
完整的Student类:
class Student{ private String name; private int age; private double height; @Override public boolean equals(Object obj) { //使用==操作符检查“参数是否为这个对象的引用” if(this==obj) return true; //使用instanceof操作符检查“参数是否为正确的类型” if(!(obj instanceof Student)) return false; //把参数转换成正确的类型。 Student student=(Student) obj; //对于该类中的每个“关键”阈,检查参数中的阈是否与该对象中对应的阈相匹配。 return this.name==student.name && this.age==student.age && this.height==student.height; } @Override public int hashCode() { //初始化 int result=17; //String类型 result=this.name.hashCode()+result; //int类型 result=this.age+result; //double类型 long heightBits=Double.doubleToLongBits(height); int heightTemp=(int)(heightBits ^ (heightBits >>> 32)); result=heightTemp+result; //返回 return result; } }
原文:http://www.cnblogs.com/lyajs/p/5796865.html