Java覆盖equals方法

前言:

  • equals方法是一个Object方法
  • 覆盖equals方法的代价比较高,最简单的方法就是不去覆盖equals方法
  • 如果不覆盖equals方法则类的每个实例只和自身相等

等价关系:

  • 自反性:对于任何非null的引用x,x.equals(x)必须返回true。
  • 对称性:对于任何非null的引用值x和y,当且仅当y.equals(x)返回true时,x.equals(y)必须返回true。
  • 传递性:对于任何非null的引用值x、y和z,如果x.equals(y)返回true,并且y.equals(z)也返回true,那么x.equals(z)也必须返回true。
  • 一致性:对于任何非null的引用值x和y,只要equals的比较操作在对象的所用信息没有改变,多次调用x.equals(y),就会一致地返回true,或者一致地返回false。
  • 非空性:对于任何非null的引用值x,x.equals(null)必须返回false。

实现高质量的equals方法的诀窍

  • 使用==操作符检查“参数是否为这个对象的引用”
  • 使用instanceof操作符检查“参数是否是正确的类型”
  • 使用参数转换为正确的类型
  • 对于该类中的每个“关键”域,检查参数中的域是否与该对象中对应的域相匹配

示例代码:

class MyPoints{
    private int x;
    private int y;
    public MyPoints(int x,int y ){
        this.x=x;
        this.y=y;
    }
    public int getX(){
        return this.x;
    }
    public int getY(){
        return this.y;
    }
    @Override
    public boolean equals(Object o){
        if(o==this)//使用==操作符检查“参数是否为这个对象的引用”
            return true;
        if(!(o instanceof MyPoints))//使用instanceof操作符检查“参数是否是正确的类型”
            return false;
        MyPoints mypoints=(MyPoints)o;// 使用参数转换为正确的类型
        return this.x==mypoints.getX()&&this.y==mypoints.getY();//对于该类中的每个“关键”域,检查参数中的域是否与该对象中对应的域相匹配
    }
}
public class equalsTest{

    public static void main(String[]args){
        MyPoints point1=new MyPoints(1,2);
        MyPoints point2=new MyPoints(2,3);
        MyPoints point3=new MyPoints(1,2);
        System.out.println("point1 equals point2 ? "+point1.equals(point2));
        System.out.println("point2 equals point1 ? "+point2.equals(point1));
        System.out.println("point1 equals point3 ? "+point1.equals(point3));
        System.out.println("point3 equals point1 ? "+point3.equals(point1));
        System.out.println("point1 equals point1 ? "+point1.equals(point1));
    }
}

输出结果:

这里写图片描述

hashcode问题

将主方法改为:

public class equalsTest{

    public static void main(String[]args){
        MyPoints point1=new MyPoints(1,2);
        MyPoints point2=new MyPoints(2,3);
        MyPoints point3=new MyPoints(1,2);
        System.out.println("point1 equals point2 ? "+point1.equals(point2));
        System.out.println("point2 equals point1 ? "+point2.equals(point1));
        System.out.println("point1 equals point3 ? "+point1.equals(point3));
        System.out.println("point3 equals point1 ? "+point3.equals(point1));
        System.out.println("point1 equals point1 ? "+point1.equals(point1));
        Map<MyPoints, String> Mapabc
            = new HashMap<MyPoints, String>();  
        Mapabc.put(point1, "point1");  
        System.out.println(Mapabc.get(new MyPoints(1,2)));
    }
}

此时期待的输出结果是,point1
但是实际上却输出了null

原因:由于MyPoints类没有覆盖hashcode方法,导致了相等的实例对象具有不同的hashcode。
解决方法:覆盖MyPoints中的hashcode方法
在MyPoints中添加代码

 @Override 
    public int hashCode() {  
       int result = hashCode;  
       if(result == 0) {  
            result = 17; //这个值是任意的 
            result = 31 * result + this.x; //31这个数字可以使用在jvm内部可以使用移位运算来代替减法来代替乘法。
            result = 31 * result + this.y;  
            hashCode = result;  
          }  
          return result;  
      } 

此时保证了equals方法相等的实例对象hashcode也是相同的。
结果截图:
这里写图片描述

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java中,equals()是一个用于比较两个对象是否相等的方法,其返回值为boolean类型。默认情况下,equals()使用的是对象的内存地址进行比较,即判断两个对象是否指向同一个内存地址。 如果我们需要对自定义对象进行比较,则需要重写equals()方法,并根据我们自己的需求来确定两个对象是否相等。一般情况下,我们需要比较对象的内容是否相等,因此我们需要在equals()方法覆盖默认的实现,实现自定义的比较方式。 下面是一个示例: ```java public class Person { private String name; private int age; public Person(String name, int age) { this.name = name; this.age = age; } @Override public boolean equals(Object obj) { if (obj == null) { return false; } if (this == obj) { return true; } if (this.getClass() != obj.getClass()) { return false; } Person other = (Person) obj; return this.name.equals(other.name) && this.age == other.age; } } ``` 在上面的例子中,我们重写了equals()方法,使其比较两个Person对象的name和age属性是否相等。在比较过程中,我们首先判断两个对象是否为同一对象或者是否为同一种类型的对象,然后再比较属性是否相等。 当我们需要比较两个对象是否相等时,可以使用equals()方法来进行比较。例如: ```java Person p1 = new Person("Tom", 18); Person p2 = new Person("Tom", 18); if (p1.equals(p2)) { System.out.println("p1和p2相等"); } else { System.out.println("p1和p2不相等"); } ``` 上述代码中,p1和p2虽然是两个不同的Person对象,但是它们的name和age属性相等,因此调用equals()方法后返回true,输出结果为“p1和p2相等”。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值