这篇文章主要讲解了“java中equals和hashCode的作用是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“java中equals和hashCode的作用是什么”吧!
什么时候覆盖equals方法我就不再说了,相信熟悉Java的读者肯定都知道,我们着重聊一下如何书写规范的equls方法。因为一个不规范的equals方法将会造成集合无法表现出预期的行为。
重写equals方法需要保证equals满足以下特性:
该要求说明对象必须等于自身,如果违背这一条你会发下集合的contains方法无法告诉你正确的结果。
对称性要求两个对象对于它们是否相等要保持一致,下面这段代码违反了对称性
public class IgnoreCaseString {
private final String s;
public IgnoreCaseString(String s) {
this.s = s;
}
@Override
public boolean equals(Object obj) {
if (obj instanceof IgnoreCaseString) {
return s.equalsIgnoreCase(((IgnoreCaseString) obj).s);
}
if (obj instanceof String) {
return s.equalsIgnoreCase((String) obj);
}
return false;
}
public static void main(String[] args) {
IgnoreCaseString ignoreCaseString = new IgnoreCaseString("Phone");
String string = "phone";
System.out.println(ignoreCaseString.equals(string));
System.out.println(string.equals(ignoreCaseString));
System.out.println("------------------------------");
List<String> list = new ArrayList<>();
list.add(string);
System.out.println(list.contains(ignoreCaseString));
}
}
这种情况通常发生在具有父子关系的对象中,子类增加的信息会影响到equals的比较结果。解决这种问题通常有两种方式,一种是通过getClass()的方式(具体的大家可以通过阅读Effective Java这本书),还有一种是在我们扩展类的功能时尽量使用复合而并不是使用继承,通过复合组件里面的域的比较也可以解决。
一致性要求如果两个对象相等,那么他们就必须保持相等,除非它们中有对象被修改了。
非空性要求所有的对象不等于null
在每个覆盖了equals方法的类中,必须重写hashCode方法。如果不这样做,会导致该类无法与所有基于散列的集合一起正常运作。hashCode方法需要遵循以下规定:
使用非0初始值的原因是让域初始值为0的那些域可以影响到散列值,如果使用0那么散列值不会再受这些域的影响,从而增加Hash冲突的可能性,从而降低了散列表的性能
31是一个奇素数,在Java中如果两个比较大的数相乘则会发生移除,31并不算一个比较大的数,其次选用31的理由是可以用移位和减法来代替乘法,31 * i = (i << 5) - i,现在的VM都可以自动完成这种优化,因此可以获得很好的性能提升
如果一个类是不可变的并且计算散列码的成本比较大,可以考虑把散列码缓存在对象内部,而不是每次请求时都重新计算(这一点在Kafka中也有所应用)。
感谢各位的阅读,以上就是“java中equals和hashCode的作用是什么”的内容了,经过本文的学习后,相信大家对java中equals和hashCode的作用是什么这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是亿速云,小编将为大家推送更多相关知识点的文章,欢迎关注!
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。