文档章节

重写equals方法

wf王帆
 wf王帆
发布于 2016/08/10 22:32
字数 864
阅读 21
收藏 0
点赞 0
评论 0

我们都知道,==是用来比较引用的(物理上的相等),而equals方法是用来比较值的(逻辑上的相等),在许多时候需要重写equals方法来实现我们的需求,比如把对象放到容器中,然后去查找对象。

在重写equals 方法时要遵循一些契约:

  • 自反性:对于非空引用x而言,x.equals(x) 必须为true
  • 对称性:对于非空引用x和y,如果x.equals(y)为true,那么y.equals(x)必须也为true
  • 传递性:对于非空引用x,y和z,如果x.equals(y)为true,并且y.equals(z)为true,那么x.equals(z)必须也为true
  • 持久性:对于非空引用x和y,如果x.equals(y) 为true,那么当在x和y并未被修改的情况下,无论执行多少次x.equals(y),这个结果都为true
  • null-false: 对于非空引用x,x.equals(null) 始终为false

 

反例:

破坏自反性

复制代码

1 public class AntiReflexive {
 2 
 3     private String str;
 4     public String getStr() {
 5         return str;
 6     }
 7 
 8     public void setStr(String str) {
 9         this.str = str;
10     }
11     
12     public boolean equals(Object o) {
13         if (!(o instanceof AntiReflexive)) {
14             return false;
15         }
16         
17         AntiReflexive ar = (AntiReflexive)o;
18         return this.str.equals(ar.str.toLowerCase());   // tolowercase 导致了自反性的破坏
19     }
20     
21     public static void main(String[] args) {
22         AntiReflexive ar1 = new AntiReflexive();
23         ar1.setStr("Hello world");
24         
25         System.out.println(ar1.equals(ar1));
26     }
27 }

复制代码

 

破坏对称性

复制代码

1 /**
 2  * 代码来自  effective java
 3  */
 4 public class AntiSymmetric {
 5 
 6     public static void main(String[] args) {
 7         CaseInsensitiveString cis = new CaseInsensitiveString("abc");
 8         String str = "abc";
 9         System.out.println(cis.equals(str));
10         System.out.println(str.equals(cis));
11     }
12 }
13 
14 class CaseInsensitiveString {
15 
16     private String s;
17     
18     public CaseInsensitiveString(String s) {
19         this.s = s;
20     }
21     
22     public boolean equals(Object o) {
23         if (o instanceof CaseInsensitiveString) {
24             return s.equalsIgnoreCase(((CaseInsensitiveString)o).s);
25         }
26         
27         if (o instanceof String) {        // 这个地方破坏了对称性,因为在String中不存在这样的逻辑,单向的
28             return s.equalsIgnoreCase((String)o);
29         }
30         
31         return false;
32     }
33 }

复制代码

 

破坏传递性

复制代码

1 public class AntiTransitive {
 2 
 3     public static void main(String[] args) {
 4         ColorPoint x = new ColorPoint(10,  20, Color.red);
 5         Point y = new Point(10, 20);
 6         ColorPoint z = new ColorPoint(10,  20, Color.blue);
 7         
 8         System.out.println("x.equals(y) = " + x.equals(y));
 9         System.out.println("y.equals(z) = " + y.equals(z));
10         System.out.println("x.equals(z) = " + x.equals(z));
11     }
12 }
13 
14 class Point {
15     private int x;
16     private int y;
17     
18     public Point(int x, int y) {
19         this.x = x;
20         this.y = y;
21     }
22     
23     public boolean equals(Object o) {
24         if (!(o instanceof Point)) {
25             return false;
26         }
27         
28         Point p = (Point)o;
29         
30         return x == p.x && y ==  p.y;
31     }
32 }
33 
34 class ColorPoint extends Point {
35     private Color color;
36     
37     public ColorPoint(int x, int y, Color color) {
38         super(x, y);
39         this.color = color;
40     }
41     
42     public boolean equals(Object o) {
43         if (!(o instanceof Point)) {
44             return false;
45         }
46         
47         if (!(o instanceof ColorPoint)) {
48             return o.equals(this);            // 这个地方破坏了传递性
49         }
50         
51         ColorPoint cp = (ColorPoint)o;
52         
53         return super.equals(o) && cp.color == this.color;
54     }
55 }

复制代码

 

在编写equals代码时,一些好的习惯是非常必要的

  • 使用 == 判断对象是否就等于this,这样就保证了自反性,还提高了performance
  • 使用instanceof 来判断这个对象是不是正确的类型,如果不是,就返回false
  • 将参数强制转换为正确的类型,因为经过了instanceof的测试,索引放心的转吧
  • 优先比较导致返回false的可能性比较大的字段,只要发现不一样就返回false
  • 对于float和double类型的字段,使用Float.compare 或者 Double.compare 方法来比较
  • 对于数组字段,遍历数组比较其中的每一项
  • 不要比较冗余字段(比如多边形类型的面积字段,就不用比较) 
  • 如果重写了equals,请重写hashcode

© 著作权归作者所有

共有 人打赏支持
wf王帆
粉丝 6
博文 64
码字总数 9386
作品 0
西安
Java 中的hashCode()和equals()

此两个方法是从Object 类继承而来的,大多数时间我们都要重写这两个方法,以下是它们之间的说明: 从Object 继承过来hasCode()取得的是当前对象的内存地址的int 映射(将内存地址映射为一个int...

idoz ⋅ 2011/12/10 ⋅ 0

重写equals和hashCode方法的示例

如果一个类有自己特有的“逻辑相等”,且需要以此进行比较时,那么就需要重写equals方法。 在Object的规范中,重写equals方法有下面几条通用约定: 自反性。 x.equals(x) == true 对称性。i...

囚徒困境 ⋅ 2015/03/01 ⋅ 0

hashcode和equals

若两个对象equals相等(重写了equals方法),但不在一个区间(没有重写hashcode方法),因为hashCode的值在重写之前是对内存地址计算得出,所以根本没有机会进行比较,会被认为是不同的对象。所...

WJobs ⋅ 06/15 ⋅ 0

java的equals重写,

java的equals方法一般情况下需要重写,以保证能够比较两个实例对象是否一致,注意重写equals方法时还要重写hashCode方法。 equals方法的注意事项:1)此方法应为继承自Object的方法(可用@Ov...

乔三爷 ⋅ 2014/06/06 ⋅ 0

Java 中正确使用 hashCode 和 equals 方法

在这篇文章中,我将告诉大家我对hashCode和equals方法的理解。我将讨论他们的默认实现,以及如何正确的重写他们。我也将使用Apache Commons提供的工具包做一个实现。 目录: hashCode()和equ...

王振威 ⋅ 2012/10/21 ⋅ 43

我理解的== 、equals 、hashcode

先来回顾一下 == 与equals 的区别 ==是运算符 java中的数据类型,可分为两类: 1.基本数据类型,也称原始数据类型。byte,short,char,int,long,float,double,boolean ,他们之间的比较,应该用...

CTO搬砖达人 ⋅ 2017/03/10 ⋅ 0

equals与”==”的区别

equals()和“==”操作用于对象的比较,检查俩对象的相等性,但是他们俩的主要区别在于前者是方法后者是操作符。由于java不支持操作符重载(overloading),“==”的行为对于每个对象来说与equ...

YoungBoyG ⋅ 2016/03/23 ⋅ 0

equals与”==”的区别

equals()和“==”操作用于对象的比较,检查俩对象的相等性,但是他们俩的主要区别在于前者是方法后者是操作符。由于java不支持操作符重载(overloading),“==”的行为对于每个对象来说与equ...

LCZ777 ⋅ 2014/03/30 ⋅ 0

equals和hashCode的区别和联系

一、前言 前段时间使用list.remove(obj)的时候重写了obj的equals方法,因为list的remove是以equals来判断标准的。但是,今天被公司的代码扫描工具提示未重写hashCode方法!!之前准备面试时也...

叫我宫城大人 ⋅ 2017/08/23 ⋅ 0

细说equals()方法和hashCode()方法

一、前言 对于这两个方法的研究,源于一道比较经典的面试题:“x.equals(y)==true;x,y可有不同的hashcode对吗?”,其实这道题的关键在于考我们对equals()方法和hashCode()方法的理解,网上看...

EnjoyAndroid ⋅ 2017/11/24 ⋅ 0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

JVM堆的理解

在JVM中,我们经常提到的就是堆了,堆确实很重要,其实,除了堆之外,还有几个重要的模块,看下图: 大 多数情况下,我们并不需要关心JVM的底层,但是如果了解它的话,对于我们系统调优是非常...

不羁之后 ⋅ 昨天 ⋅ 0

推荐:并发情况下:Java HashMap 形成死循环的原因

在淘宝内网里看到同事发了贴说了一个CPU被100%的线上故障,并且这个事发生了很多次,原因是在Java语言在并发情况下使用HashMap造成Race Condition,从而导致死循环。这个事情我4、5年前也经历...

码代码的小司机 ⋅ 昨天 ⋅ 1

聊聊spring cloud gateway的RetryGatewayFilter

序 本文主要研究一下spring cloud gateway的RetryGatewayFilter GatewayAutoConfiguration spring-cloud-gateway-core-2.0.0.RC2-sources.jar!/org/springframework/cloud/gateway/config/G......

go4it ⋅ 昨天 ⋅ 0

创建新用户和授予MySQL中的权限教程

导读 MySQL是一个开源数据库管理软件,可帮助用户存储,组织和以后检索数据。 它有多种选项来授予特定用户在表和数据库中的细微的权限 - 本教程将简要介绍一些选项。 如何创建新用户 在MySQL...

问题终结者 ⋅ 昨天 ⋅ 0

android -------- 颜色的半透明效果配置

最近有朋友问我 Android 背景颜色的半透明效果配置,我网上看资料,总结了一下, 开发中也是常常遇到的,所以来写篇博客 常用的颜色值格式有: RGB ARGB RRGGBB AARRGGBB 这4种 透明度 透明度...

切切歆语 ⋅ 昨天 ⋅ 0

CentOS开机启动subversion

建立自启动脚本: vim /etc/init.d/subversion 输入如下内容: #!/bin/bash## subversion startup script for the server## chkconfig: 2345 90 10# description: start the subve......

随风而飘 ⋅ 昨天 ⋅ 0

Nginx + uwsgi @ubuntu

uwsgi 安装 sudo apt-get install python3-pip # 注意 ubuntu python3默认没有安装pippython3 -m pip install uwsgi 代码(test.py) def application(env, start_response): start_res......

袁祾 ⋅ 昨天 ⋅ 0

版本控制工具

CSV , SVN , GIT ,VSS

颖伙虫 ⋅ 昨天 ⋅ 0

【2018.06.19学习笔记】【linux高级知识 13.1-13.3】

13.1 设置更改root密码 13.2 连接mysql 13.3 mysql常用命令

lgsxp ⋅ 昨天 ⋅ 0

LVM

LVM: 硬盘划分分区成物理卷->物理卷组成卷组->卷组划分逻辑分区。 1.磁盘分区: fdisk /dev/sdb 划分几个主分区 输入t更改每个分区类型为8e(LVM) 使用partprobe生成分区的文件:如/dev/sd...

ZHENG-JY ⋅ 昨天 ⋅ 0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部