第八条:覆盖equals时请遵守通用约定
==是物理相等
equals是逻辑相等
因为每个类的实例对象本质上都是唯一的 ,利用物理相等(==)是指一个实例只能相等于它自己。
利用逻辑相等是(equals)指 一个实例是否和另一个实例的某些关键域相等,从而来判断这两实例是否相等。
Object类的equals方法的实现:物理相等的话就逻辑相等。
什么时候不需要覆盖Object类的equals方法?
1.希望这个类的实例只能和自身相等,不覆盖equals方法,只继承Object类的equals方法。
我们比较这个类的实例对象是否相等,无论是使用 == 或者equals 实际上都是 进行物理相等的判断。
2.这个类的超类已经覆盖了Object类的equals方法,并且这个类继承超类的equals方法,对于它自身来说也是合适的,这时不覆盖equals方法。
例如:Set实现类都继承了AbstractSet类的equals方法,List实现类都继承了AbstractList类的equals方法,Map实现类都继承了AbstractMap类
的equals方法。
3.确定这个类的equals方法永远不会被调用时。
什么时候需要覆盖Object类的equals方法?
当一个类需要自己的特有的“逻辑相等”时,例如我们希望两个实例的某些关键域相等时,我们就认为它们两者是逻辑上相等的,调用equals方法就返回true。
如果不覆盖equals方法,继承Object类的equals方法,实际上进行的物理相等的判断,即一个实例只能和它自身相等。
例如:一些“值类”,如Integer,Date,String等,这些类我们在调用它们的equals方法时,希望只要两个实例的关键域相等,就返回true,而不是两个实例
是同一个实例时再返回true。而且这些类的equals方法必须被覆盖,从而在作为Map集合的key,或者Set集合的元素时出现符合预期的结果,因为它们的equals
方法会被调用来判断两个实例是否逻辑相等。
但是对于一些实例受控的类,如单例类,枚举类,这些类就不需要覆盖equals方法了。
正确覆盖equals方法,需要遵守它的通用约定:
1.自反性: x.equals(x) 必须返回true 。实例自身必然逻辑相等。
2.对称性: x.equals(y) 与 y.equals(x) 返回结果应该相同,同为true或者同为false。
3.传递性: x.equals(y)返回true,y.equals(z)返回true,则x.equals(z)应该返回true。
4.一致性: 只要比较的实例对象的关键属性值没有改变 ,那么无论调用多少次equals方法返回的结果都应该相同,一致。
5.对于非null的x实例,x.equals(null) 永远返回false。
上面的五条约定,十分重要。有许多类,包括所有的集合类在内,都依赖于传递给它们的实例对象是否遵守了equals约定。
结合所有的约定需求,得到实现高质量equals方法的诀窍:
1.使用==操作符检查“参数对象的引用是否是调用对象”的相同引用。如果是,返回true。
2.使用instanceof操作符检查参数对象的类型是否正确。这里的类型正确,一般是指验证参数对象的类型是否是当前类。有些情况
也可以在当前类实现的接口的其他实现类,之间进行比较。
3.把参数对象转化为正确的类型。因为已经经过instanceof类型检查过了,类型转化可以确保成功。
4.对于该类中的每个“关键”域,检查参数对象中域是否与当前对象中对应的域相匹配。
如果上面的测试全部成功,就返回true,否则就返回false。
其中对于既不是float也不是double类型的基本类型域,可以直接使用==操作符进行比较,对于float类型的域,使用Float.compare方法,对于
double类型的域,使用Double.compare方法。对于引用类型的域,调用引用类型的equals方法进行比较。
当一个类的equals方法完成时,应该明确:它是否是对称的,传递的,一致的? 应该写测试代码去进行测试。
还要铭记一下告诫:
1.覆盖equals方法时总要覆盖hashCode方法。(第九条)
2.不要企图让equals方法太过智能。
3.不要把参数Object类型 换成任何其他类型。因为Object类提供的equals方法的参数类型就是Object,如果把参数Object类型换成其他类型,
就变成了equals方法的重载,而不是equals方法的重写。 所以可以添加@Override注解 来避免这种错误。
第八条:覆盖equals时请遵守通用约定的更多相关文章
- 第8条:覆盖equals时请遵守通用约定
第8条:覆盖equals时请遵守通用约定 引言:尽管Object是一个具体类,但是设计它主要是为了拓展.它所有的非final方法(equals.hashCode.toString.clone和fina ...
- Item 8 覆盖equals时请遵守通用约定
在覆盖equals方法的时候,你必须要遵守它的通用约定,不遵守,写出来的方法,会出现逻辑错误.下面是约定的内容: equals方法实现了等价关系: 自反性.对于任何非null的引用值,x.eq ...
- 覆盖equals时请遵守通用约定
Object类中非final修饰的方法有equals().hashCode().toString().finalize().clone()1.equals()方法不需要被覆盖的情况:1)实例化的对象只 ...
- 【Effective Java】4、覆盖equals时请遵守通用约定
package cn.xf.cp.ch02.item8.transitivity; public class Point { private final int x; private final in ...
- 第10项:重写equals时请遵守通用约定
重写equals方法看起来似乎很简单,但是有许多重写方式会导致错误,而且后果非常严重.最容易避免这类问题的办法就是不覆盖equals方法,在这种情况下,类的每个实例都只能与它自身相等.如果满足了以 ...
- 覆盖equals方法时请遵守通用约定
覆盖equals方法时请遵守通用约定 覆盖equals方法看起来很简单,但是有许多覆盖方式会导致错误,并且后果很严重.最容易避免这种类问题的方法就是不覆盖equals方法,在这种情况下,类的每个实 ...
- EffectiveJava(8)覆盖equals是要遵守的约定
覆盖equals是要遵守的约定 1.覆盖种类: -类的每个1实例本质上都是唯一的 -不关心类是否提供了"逻辑相等"的测试功能(Random测试是否能随机相同数字) -超类已经覆盖了 ...
- 重写equals时,遵守的规定
0 正确的equals方法 public class MyClass { // 主要属性1 private int primaryAttr1; // 主要属性2 private int prima ...
- 第8条:覆盖equals时遵守通用约定
如果不需要覆盖equals方法,那么就无需担心覆盖equals方法导致的错误. 什么时候不需要覆盖equals方法? 1.类的每个实例本质上是唯一的. 例如对于Thread,Object提供的equa ...
随机推荐
- [Luogu4149][IOI2011]Race
BZOJ权限题qwq Luogu sol 树上路径当然是淀粉质辣! 考虑所有过重心的路径.开一个\(10^6\)大小的数组\(t\)表示某一路径长度的最小边数,初始化为\(inf(i>0)\), ...
- 使用 vscode将本地项目上传到github以及删除github上的某个文件夹
安装Git后,可以看到windows环境下有两个命令输入窗口Git CMD 和Git Bash Git GUI是可视化图形界面 Git中的Bash是基于CMD的,在CMD的基础上增添一些新的命令与功能 ...
- maven工程 java 实现文件上传 SSM ajax异步请求上传
java ssm框架实现文件上传 实现:单文件上传.多文件上传(单选和多选),并且用 ajax 异步刷新,在当前界面显示上传的文件 首先springmvc的配置文件要配置上传文件解析器: <!- ...
- 索信达携手8Manage,打造项目管理系统信息化体系
[导语]金融大数据已逐渐成为行业潮流,作为金融大数据应用提供商,深圳索信达企业为了实现业务和研发项目的多重管理需求,决定引入8Manage项目管理系统,提高项目管控能力和工作效率,从而提高企业的核心竞 ...
- angular路由模块(二)
上一章写的是如何创建一个简单的路由,这一样我们来看看如何创建一个路由模块.angular的思想就是(模块,组件,子组件.....). 我们在src/app目录下创建一个跟路由模块app-routing ...
- 深入java虚拟机学习 -- 类的加载机制(三)
类的初始化时机 在上篇文章中讲到了类的六种主动使用方式,反射是其中的一种(Class.forName("com.jack.test")),这里需要注意一点:当调用ClasLoade ...
- MapReduce并行编程模型和框架
传统的串行处理方式 有四组文本数据: "the weather is good", "today is good", "good weather is ...
- 【python学习笔记】10.充电时刻
[python学习笔记]10.充电时刻 任何python都可以作为模块倒入 *.pyc:平台无关的经过编译的的python文件, 模块在第一次导入到程序中时被执行,包括定义类,函数,变量,执行语句 可 ...
- Mybatis+Mysql插入数据库返回自增主键id值的三种方法
一.场景: 插入数据库的值需要立即得到返回的主键id进行下一步程序操作 二.解决方法: 第一种:使用通用mapper的插入方法 Mapper.insertSelective(record): 此方法: ...
- redis分片和哨兵
1 Redis的使用 1.1 Redis入门案例 1.1.1 什么样的数据使用缓存 说明:使用缓存其实为了减少用户查询数据库的时间.如果数据频繁的变更.不适用缓存.缓存中的数据应该保存修改频率不高的数 ...