一、equals方法介绍

1.1.通过下面的例子掌握equals的用法

 1 package cn.galc.test;2 3 public class TestEquals {4     public static void main(String[] args) {5         /**6          * 这里使用构造方法Cat()在堆内存里面new出了两只猫,7          * 这两只猫的color,weight,height都是一样的,8          * 但c1和c2却永远不会相等,这是因为c1和c2分别为堆内存里面两只猫的引用对象,9          * 里面装着可以找到这两只猫的地址,但由于两只猫在堆内存里面存储在两个不同的空间里面,
10          * 所以c1和c2分别装着不同的地址,因此c1和c2永远不会相等。
11          */
12         Cat c1 = new Cat(1, 1, 1);
13         Cat c2 = new Cat(1, 1, 1);
14         System.out.println("c1==c2的结果是:"+(c1==c2));//false
15         System.out.println("c1.equals(c2)的结果是:"+c1.equals(c2));//false
16     }
17 }
18
19 class Cat {
20     int color, weight, height;
21
22     public Cat(int color, int weight, int height) {
23         this.color = color;
24         this.weight = weight;
25         this.height = height;
26     }
27 }

1.2.画出内存分析图分析c1和c2比较的结果

程序:

Cat c1 = new Cat(1,1,1);

Cat c2 = new Cat(1,1,1);

执行完之后内存之中的布局如下图所示,

  c1指向一个对象,c2也指向一个对象,c1和c2里面装着的是这两只Cat对象在堆内存里面存储的地址,由于这两只Cat对象分别位于不同的存储空间,因此c1和c2里面装着的地址肯定不相等,因此c1和c2这两个引用对象也肯定不相等。因此执行:“System.out.println(c1==c2);”打印出来的结果肯定是false。因此你new出来了两个对象,你放心,这两个对象的引用永远不一样,一样的话就会把其中一个给覆盖掉了,这个可不成。c1是不是等于c2比较的是c1和c2这两个引用里面装着的内容,因为new出来的两个对象的它们的引用永远不一样,因此c1和c2这两个引用的内容也永远不一样,因此c1永远不可能等于c2。因此通过比较两个对象的引用是永远无法使得两个对象相等的,一模一样的。

  要想判断两个对象是否相等,不能通过比较两个对象的引用是否相等,这是永远都得不到相等的结果的,因为两个对象的引用永远不会相等,所以正确的比较方法是直接比较这两个对象,比较这两个对象的实质是不是一样的,即这两个对象里面的内容是不是相同的,通过比较这两个对象的属性值是否相同而决定这两个对象是否相等。

  Object类提供了一个equals()方法来比较两个对象的内容是否相同,因此我们可以采用这个方法去比较两个对象是否在逻辑上“相等”。如:c1.equals(c2);这里是调用从Object类继承下来的equals()方法,通过查阅API文档得到Object类里的equals方法的定义如下:

public boolean equals(Object obj)

  在Object这个类里面提供的Equals()方法默认的实现是比较当前对象的引用和你要比较的那个引用它们指向的是否是同一个对象,即和“c1==c2”这种写法是一样的,“c1.equals(c2)”与“c1==c2”是完全等价的。因此直接使用继承下来的equals()方法也是无法直接比较两个对象的内容是否相同的,为此,我们必须得重写equals()方法,改变这个方法默认的实现。

下面在Cat类里面重写这个继承下来的equals()方法

 1 class Cat {2     int color, weight, height;3 4     public Cat(int color, int weight, int height) {5         this.color = color;6         this.weight = weight;7         this.height = height;8     }9
10     /**
11      * 这里是重写相等从Object类继承下来的equals()方法,改变这个方法默认的实现,
12      * 通过我们自己定义的实现来判断决定两个对象在逻辑上是否相等。
13      * 这里我们定义如果两只猫的color,weight,height都相同,
14      * 那么我们就认为这两只猫在逻辑上是一模一样的,即这两只猫是“相等”的。
15      */
16     public boolean equals(Object obj){
17         if (obj==null){
18             return false;
19         }
20         else{
21             /**
22              * instanceof是对象运算符。
23              * 对象运算符用来测定一个对象是否属于某个指定类或指定的子类的实例。
24              * 对象运算符是一个组合单词instanceof。
25              * 该运算符是一个双目运算符,其左边的表达式是一个对象,右边的表达式是一个类,
26              * 如果左边的对象是右边的类创建的对象,则运算结果为true,否则为false。
27              */
28             if (obj instanceof Cat){
29                 Cat c = (Cat)obj;
30                 if (c.color==this.color && c.weight==this.weight && c.height==this.height){
31                     return true;
32                 }
33             }
34         }
35         return false;
36     }
37 }

此时在再main方法里面执行打印的命令:

 1 public static void main(String[] args) {2         /**3          * 这里使用构造方法Cat()在堆内存里面new出了两只猫,4          * 这两只猫的color,weight,height都是一样的,5          * 但c1和c2却永远不会相等,这是因为c1和c2分别为堆内存里面两只猫的引用对象,6          * 里面装着可以找到这两只猫的地址,但由于两只猫在堆内存里面存储在两个不同的空间里面,7          * 所以c1和c2分别装着不同的地址,因此c1和c2永远不会相等。8          */9         Cat c1 = new Cat(1, 1, 1);
10         Cat c2 = new Cat(1, 1, 1);
11         System.out.println("c1==c2的结果是:"+(c1==c2));//false
12         System.out.println("c1.equals(c2)的结果是:"+c1.equals(c2));//true
13     }

  这一次得到的结果就与上次没有重写equals()方法时得到的结果就不一样了:

  “System.out.println(c1 == c2);”打印出来的结果依然是false,因为这里是比较两个对象的引用里面的内容,这两个引用里面的内容当然不相等,而且永远不会相等,所以打印出来的结果肯定是false。

  “System.out.println(c1.equals(c2));”打印出来的结果为true,因为我们在Cat类里面重写了equals()方法,改变了这个方法默认的实现,我们把方法的实现改为只要这个两个对象是真的存在,并且都是猫,并且它们的颜色(color),身高(height)和体重(weight)都相同,那么这两只猫在逻辑上就是一模一样的,是完全相同的两只猫,即这两只猫是“相等”的。所以这里打印出来的结果是true。

1.3.如何比较两个字符串对象是否相等?

看下面的例子:

1 public class TestEquals {
2
3     public static void main(String args[]){
4         String s1 = new String("hello");
5         String s2 = new String("hello");
6         System.out.println("s1 == s2的结果是:"+(s1 == s2));//false
7         System.out.println("s1.equals(s2)的结果是:"+s1.equals(s2));//true
8     }
9 }

这一次是比较两个字符串对象是否相等:

  System.out.println(s1 == s2);

  打印出来的结果依然是fase,因为这里比较的是s1和s2两个字符串对象的引用,两个对象的引用永远不会相等,所以打印出来的结果为false。

  System.out.println(s1.equals(s2));

  打印出来的结果为true,因为在String类里面重写了从Object类继承(所有的类都是从Object类继承下来,String类当然也不例外,从父类继承下来就拥有了父类的一切属性与方法,所以Sting类里面也有equals()方法,并且还把这个继承下来的equals()方法重写了)下来的equals()方法,改变了这个方法默认的实现,

  在String类里面是这样重写equals()方法的实现的:用当前的这个字符串对象和指定的字符串对象比较,指定的字符串对象不能为空并且这个对象的字符序列和当前这个字符串对象的字符串序列一样,如果这些条件都满足,那么这两个字符串对象就是相等的。

因此这里的s2已经满足了条件,所以打印出来的结果是true。

  以后在某一个类里面比较两个对象是否相等时,首先去API文档里面查找这个类是否重写了从Object类继承下来的equals()方法。如果重写了equals()方法,那么在比较两个对象是否相等时调用的就是重写以后的equals()方法,如果没有重写,那么调用时就是直接调用从Object类里面的继承下来的那个equals()方法,并且采用equals()方法默认的实现去比较两个对象是否相等。因此每一个类都可以根据需要对从Object类继承下来的equals()方法进行重写。

  对于在API文档里面找某个类,如果一个类不用引入包就可以直接使用,那么这个类肯定是在java.lang这个包里面,如这里的String类,直接就可以使用了,所以String类一定是在java.lang这个包里面。使用某个类时看这个类引入的是哪个包,然后就去这个包里面找这个类,不用引入包的类一定是位于java.lang里面,直接去java.lang里面找就可以了。

  总结:比较两个对象是否相等,我们采用equals()方法,判断两个对象是否相等的条件是由我们重写equals()方法的实现后定义的,这样就可以比较灵活地使用equals()方法在不同的类里面比较位于同一类下的两个对象是否相等了。

转载于:https://www.cnblogs.com/kexianting/p/8521586.html

java中的equals方法相关推荐

  1. Java中重写equals()方法时注意点

    Java中重写equals()方法时注意点 一直说,重写一个对象的equals()方法时我们必须重写HashCode()方法,但是如果我们不重写呢?会有什么影响呢? 首先看一下,什么情况下我们需要重写 ...

  2. java中equals函数所在的类,重写Java中的equals方法介绍

    Java中,只有8种基本类型不是对象,例如:4种整形类型(byte, short, int,long),2种浮点类型(flout, double),boolean, char不是对象,其他的所有类型, ...

  3. JAVA中“==”与equals()方法区别

    equals 方法是 java.lang.Object 类的方法 有两种用法说明: (1)对于字符串变量来说,使用"=="和"equals()"方法比较字符串时 ...

  4. java中的equals方法+hashCode方法

    [0]README 0.1)以下内容均为原创,包括源代码, 旨在理清 equals 和 hashCode 方法的 实现原理: 0.2) for full resource code, please v ...

  5. JAVA中重写equals()方法的同时要重写hashcode()方法

    object对象中的 public boolean equals(Object obj),对于任何非空引用值 x 和 y,当且仅当 x 和 y 引用同一个对象时,此方法才返回 true:注意:当此方法 ...

  6. 在聊Java中的equals方法

    2019独角兽企业重金招聘Python工程师标准>>> java规范中equals方法特征 自反性(对于任何非空引用x, x.equals(x) 返回true;) 对称性(对于任何引 ...

  7. Java 中 == 和equals()方法的不同点

    在Java中,基本变量(int a=5)存储在栈中,引用型变量存储在栈和堆中,在栈中存储变量的地址,在堆中存储变量的值. ==,对于基本变量比较的是变量的值,对于引用变量比较的是变量的地址,是地址的比 ...

  8. Java中的equals()方法

    equals()在哪里 首先我们知道Java中Object类是所有类的父类,它里面定义了equals()方法: public boolean equals(Object obj) {return (t ...

  9. Java中的equals()方法重写

    public boolean equals(Object obj) {return (this == obj); } Object类提供的 equals() 方法默认是用 == 来进行比较的,也就是说 ...

最新文章

  1. ICRA 2021自动驾驶相关论文汇总 | 科研党看过来,全文干货
  2. 2019年企业新增长 从雇佣“机器人员工”开始
  3. C# 利用Jmail接收邮件
  4. 如何把你的搜索引擎也加入到Firefox中
  5. 动态数组的各种操作 0104 c#
  6. java 课后习题 计算两个日期之间的天数
  7. Spring整合Activiti工作流
  8. 树莓派3ftp服务器修改地址,树莓派3搭建ftp服务器
  9. [转]word 转换成pdf
  10. xp系统能不能安装mysql_win xp32位系统安装mysql详细步骤
  11. 【matlab】iir滤波器
  12. Flask Swagger 文档自动生成
  13. ROS里程计的学习(odometry) (二)
  14. 配置JAVA_HOME
  15. MyBatis学习第一步
  16. 安卓学习笔记3.1 线性布局
  17. 使用 ERD Online元数据管理平台,轻松创建和共享企业元数据
  18. NOJ 1003.快速排序
  19. 【SCSS】常用的SCSS语法
  20. 股票数据抓取接口文章转载

热门文章

  1. dubbo扫描第三方包_今天来浅谈一下dubbo
  2. python flask web开发_Python Flask web后端开发
  3. 驱动操作硬件的根本操作
  4. mysql查询是否用index_mysql – 为什么这个查询使用where而不是index?
  5. 如何正确使用as follows 与 following
  6. dw网页设计作品_10个富有特色的网页设计精选作品
  7. Windows系统——VMTK安装教程
  8. FPGA之道(23)VHDL的signal、variable与constant
  9. 【 FPGA 】UltraFast设计方法学笔记(时钟)
  10. 通过css类/选择器选取元素 文档结构和遍历 元素树的文档