实习日记 09/03 day42 高效Java---Equals

概览

Object是一个具体得类,设计它的目的是为了扩展,它所有的非final方法(equals、hashCode、toString、clone和finalize)都有明确的通用约定,因为他们被设计成要覆盖的,任何一个类,它覆盖这些方法的时候,都有责任遵守这些通用约定,其他依赖于这些约定的类,就无法正常工作。
在这里插入图片描述

覆盖Equals时需遵守的约定

==与Equals的区别:

基础变量:boolean、short、int、long、float、double、char、byte
在对比非基础变量类型的时候,比如String,==实际上对比的是两个对象在内存的位置信息,如果是两个不相同的对象则会产生false,即使两个对象的值相等。String类型重写了Object列的Equals方法实现对比值。

覆盖equals方法看起来比较简单,但是覆盖方式会导致错误,并且会造成严重的后果。最容易避免这种问题的方法就是不覆盖equals方法,在这种情况下,类的每个实例都只与他自身相等,如果满足了下面的任意一个条件,就是所期望结果:

  1. 类的每个实例本质上都是唯一的
  2. 不关心类是否提供了逻辑相等
  3. 超类已经覆盖了equals,从超类继承过来的行为对子类也是合适的
  4. 类是私有的或是包级私有的,可以确定它的equals方法永远不会被调用

在这些情况下不应该在尝试去覆盖equals,或者说覆盖equals也没有实际的意义,如满足1,去对比两个Thread类是否相等,因为对象具有唯一性,也没有具体值,equals比多此一举,,Thread表示的是活动,不是值比较没有任何意义,再比如设计了一个不需要考虑值的类,此时在比较也没有任何意义。第三条可以举例如Set和List都从父类AbstractSet中继承了equals的实现,直接拿来吧,你,使用即可,也不需要使用equals。。假如设计了一个类,具有自己特有的逻辑属性,此时我们就应该覆盖equals方法。
比如说设计了一个我最爱玩的英雄联盟类,定义了一个亚索对象,一个维嘉对象,这俩肯定不等,但是如果定义了一个黑夜使者亚索和西部牛仔亚索,他俩套了不同的皮,但是都是亚索,逻辑上都是哈撒鸡,所以equals应该返回True。我们在比较这种英雄类的时候,当队友选了西部牛仔,由于equals返回了true,所以你没法选择亚索这个英雄。,你是想知道是不是亚索,而无关是不是完全一样的对象,这时候要去覆盖equals。也遵循数学上的定义:自反性、对称性、传递性、一致性。

实现高质量Equals的技巧:

  1. 使用==操作符检查“参数是否是这个对象的引用”,如果是,则百分之百相等。
  2. 使用instanceof操作符监测“参数是否为正确的类型”。如果不是,则返回false
  3. 把参数转换为正确的可对比的类型
  4. 对于自定义类中的每个“关键域”,检查参数中的域是否与该对象中对应的域相匹配。
  5. 覆盖Equals一定要覆盖hashcode
  6. equals方法不能太过复杂
  7. 不要将Equals中的object对象替换为其他的类型。

覆盖equals总要覆盖hashcode

一个很常见的错误根源在于没有覆盖hashcode,在每一个覆盖了equals方法的类中,也必须覆盖hashcode方法,如果不这样做会贝贝Object.hashCode的通用约定,从而导致该类无法结合所有基于散列的集合一起正常工作。

import java.util.HashMap;

public final class PhoneNumber {
    private final short areaCode;
    private final short preFix;
    private final short lineNumber;
    public PhoneNumber(int areaCode,int preFix,int lineNumber){
        //范围审核
         rangeCheck(areaCode,999,"area code");
         rangeCheck(preFix,999,"preFix");
         rangeCheck(lineNumber,9999,"lineNumber");
         this.areaCode = (short) areaCode;
         this.preFix=(short)preFix;
         this.lineNumber=(short)lineNumber;
    }
    private static void rangeCheck(int arg,int max,String name){
        if(arg<0 || arg > max){
            throw  new IllegalArgumentException(name+": "+arg);
        }
    }
    @Override
    public boolean equals(Object o){
        if(o==this){
            return true;
        }
        if(!(o instanceof PhoneNumber)){
            return false;
        }
        PhoneNumber ph=(PhoneNumber) o;
        return ph.lineNumber==lineNumber && ph.preFix==preFix && ph.areaCode == areaCode;
    }

    public static void main(String[] args) {
        HashMap<PhoneNumber,String> hashMap=new HashMap<PhoneNumber, String>();
        hashMap.put(new PhoneNumber(707,867,5309),"逢坂大河");
        System.out.println(hashMap.get(new PhoneNumber(707,867,5309)));
    }
}

测试显示为null,由于PhoneNumber没有覆盖hashCode方法,导致两个相等的实例具有不同的散列码。

@Override
    public int hashCode(){
        int result=17;
        result = 31*result+areaCode;
        result = 31*result+preFix;
        result = 31*result+lineNumber;
        return result;
    }

在这里插入图片描述

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 覆盖和equals是面向对象编程中的两个重要概念。 覆盖指的是子类重写父类的方法,使得子类可以根据自己的需求来实现该方法。在覆盖时需要注意方法名、参数列表和返回值类型必须与父类方法一致,否则会编译错误。覆盖可以提高代码的复用性和可维护性。 equals是Object类中的一个方法,用于比较两个对象是否相等。默认情况下,equals方法比较的是两个对象的引用是否相等,即是否指向同一个内存地址。如果需要比较两个对象的内容是否相等,就需要在类中重写equals方法,并根据类的属性来比较对象的内容。在重写equals方法时,需要满足自反性、对称性、传递性和一致性等条件,否则可能会导致程序出错。 总之,覆盖和equals都是面向对象编程中非常重要的概念,掌握它们可以提高代码的质量和效率。 ### 回答2: 覆盖与equalsJava中面向对象编程的两个基础概念,它们的正确应用非常重要。 覆盖(Override)是指子类重写(覆盖)父类的方法。在Java中,子类可以重写其父类的方法,子类的方法名、参数类型和返回值类型必须与父类方法一致。通过覆盖,我们可以实现灵活的代码复用和更高效的代码运行。 在实际开发中,我们需要注意覆盖的一些规则:首先,被覆盖的方法必须有相同的方法名、参数列表和返回值类型;其次,覆盖的方法修饰符不能比被覆盖的方法更严格;最后,被覆盖的方法不能是final类型的。 equalsJava中的一个方法,它用于比较两个对象是否相等。在默认情况下,equals方法比较的是两个对象的引用地址,但是我们可以重写equals方法,来使其比较对象的内容是否相等。 当我们需要比较自定义的对象时,必须重写equals方法,因为默认的equals方法不能满足我们的需求。在自定义equals方法时,需要遵循几个规则:首先,比较对象时需要保证其一致性、反射性、对称性和传递性;其次,equals方法必须是自反的,并且不能与其他对象类型(如String类型)相等;最后,equals方法必须与hashCode方法一致。 总之,覆盖和equals方法是Java中面向对象编程的非常基础的概念,我们需要在实际开发中灵活应用,以实现更高效、更灵活和更安全的代码编写。 ### 回答3: 覆盖是指子类可以覆盖父类中的某些方法,使得子类在调用这些方法时,执行的是自己的实现而不是父类的实现。而equals()方法是用来比较两个对象是否相等的方法,它是Object类中的一个方法,在子类中也可以重写。 在覆盖中,子类通过重写父类中的某些方法,可以添加或修改方法中的行为。在Java中,子类中必须要实现父类中的抽象方法,否则子类就必须声明为抽象类。同时,子类通过super关键字可以调用父类中被覆盖的方法,从而实现在子类中添加新的行为。 在equals()方法中,如果不重写,则会默认调用Object类中的equals()方法,这种比较方式只比较两个对象的引用是否相等,而不是对象的值是否相等。因此,在比较两个对象的值时,需要在子类中重写equals()方法,并根据实际需求来实现比较两个对象的值是否相等。 在重写equals()方法时,需要遵循以下几点原则: 1. 自反性:对于任何非 null 的对象 x,x.equals(x) 必须返回 true。 2. 对称性:对于任何非 null 的对象 x 和 y,如果 x.equals(y) 返回 true,则 y.equals(x) 必须返回 true。 3. 传递性:对于任何非 null 的对象 x、y 和 z,如果 x.equals(y) 返回 true,y.equals(z) 返回 true,则 x.equals(z) 必须返回 true。 4. 一致性:对于任何非 null 的对象 x 和 y,只要对象的状态没有发生改变,则 x.equals(y) 多次调用时必须返回相同的结果。 5. 非空性:对于任何非 null 的对象 x,x.equals(null) 必须返回 false。 综上所述,覆盖和equals()方法都是非常重要的Java基础概念,对于Java程序员来说,熟练掌握这两个概念可以帮助我们更好地进行Java开发。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值