Object中equals方法的机制研究

前言

java入门时的噩梦其一,便是String类型的等值判断,相信很多新手会直接用 == 判断相等,但这是错误的,因为String是引用类型,==判断的是引用类型的内存块地址(不是直接判断物理内存地址,而是比较的物理地址进行运算后得到的一个十六进制串),而不是判断字符串本身。

下面会通过示例代码详细介绍自定义引用类型对象的等值比较。

代码示例

  1. 定义Person对象,该对象有两个属性,age属性为基础数值类型,name属性为字符串引用类型。
  2. 重写Person对象的hashCode方法,使其方法返回值只由name属性决定。
  3. 重写equals方法,该方法需要结合hashCode方法使用。hashCode使用name作为唯一判定依据,则equals方法的逻辑也应该是只判断name属性。
public class HashSetMain {
    public static void main(String[] args) {
        // 创建HashSet对象
        HashSet<Person> hs = new HashSet<>();
        // 将Person对象存入集合
        hs.add(new Person("lisa", 21));
        hs.add(new Person("lisi", 32));
        hs.add(new Person("lisi", 33));
        hs.add(new Person("leilei", 31));
        hs.add(new Person("lusi", 25));
        hs.add(new Person("lusi", 25));
        // 遍历集合中的元素
        Iterator<Person> it = hs.iterator();
        while (it.hasNext()) {
            Person p = it.next();
            System.out.println(p);
        }
    }

    static class Person {
        private String name;
        private int age;

        public Person(String name, int age) {
            this.name = name;
            this.age = age;
        }

        // 重写hashCode方法,返回name属性的哈希值
        @Override
        public int hashCode() {
            return name.hashCode();
        }

        // 重写equals方法
        // 对于该用例,有三种处理方案

        /**
         * 常用方案一:基础的String比较,
         * 先判断对象是否都为Person类型,都为Person类型时,比较两个Person对象的name属性是否相同。
         * @param obj
         * @return true表示相等;false表示不是相等
         */
//        @Override
//        public boolean equals(Object obj) {
//            if (!(obj instanceof Person)) {
//                return false;
//            }
//            return this.name.equals(((Person) obj).name);
//        }

        /**
         * 方案二:这个方案有点绕,而且因为没有做类型判定,很容易出问题,不建议使用。
         * 此是为了演示equals方法的作用
         * 1. 第一层:传入的obj是Person对象,那么obj.equals(name)会走到下面这个equals重写方法中。
         * 2. 第二层:由上一层得出,这一层obj入参实际是String类型,也就是上面的name。到这里就相当于name.equals(this.name)了,
         *           所以这种equals也能实现要求。
         * @param obj 比较对象
         * @return true表示相等;false表示不是相等
         */
//        @Override
//        public boolean equals(Object obj) {
//            return obj.equals(name);
//        }

        /**
         * 方案三:这个方案得益于jvm的字符串常量池缓存机制。
         * main方法中加入到HashSet中的Person对象的name都是字符串常量
         * 也就意味着new Person("lisi", 32)与new Person("lisi", 33)这两个对象里的name字段实际上是指向的同一个内存地址
         * 那么使用 == 自然也就会返回true
         * @param obj
         * @return true表示相等;false表示不是相等
         */
        @Override
        public boolean equals(Object obj) {
            if (!(obj instanceof Person)) {
                return false;
            }
            return this.name == ((Person) obj).name;
        }

        @Override
        public String toString() {
            return "Person{" +
                    "name='" + name + '\'' +
                    ", age=" + age +
                    '}';
        }
    }
}

通过以上三种方案,都能实现通过name属性对Person对象去重,最终的输出结果都是

运行效果

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Object类的equals方法Java的一个重要方法,用于判断两个对象是否相等。它的默认实现是比较两个对象的引用是否相同,即比较两个对象的内存地址是否相同。但是,我们可以根据需要在自定义类重写equals方法来定义自己的相等条件。 一般来说,我们在重写equals方法时需要满足以下几个要求: 1. 自反性:对于任何非空引用值x,x.equals(x)应该返回true。 2. 对称性:对于任何非空引用值x和y,如果x.equals(y)返回true,则y.equals(x)也应该返回true。 3. 传递性:对于任何非空引用值x、y和z,如果x.equals(y)返回true,并且y.equals(z)也返回true,则x.equals(z)也应该返回true。 4. 一致性:对于任何非空引用值x和y,多次调用x.equals(y)应该始终返回相同的结果,前提是对象上的信息没有被修改。 5. 非空性:对于任何非空引用值x,x.equals(null)应该返回false。 在实际使用,我们可以根据对象的属性来判断它们是否相等,比如比较两个字符串的内容是否相同,或者比较两个自定义类对象的特定属性是否相等。在重写equals方法时,通常还需要重写hashCode方法,以确保相等的对象具有相同的哈希码。 需要注意的是,equals方法在比较对象时应该使用instanceof关键字进行类型检查,以避免ClassCastException异常的发生。此外,还可以使用Objects类的equals方法来简化equals方法的实现。 这就是关于Object类的equals方法的一些基本介绍。如果你有任何进一步的问题,请随时提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Tomshidi

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值