首页 > 其他分享 >为什么重写了 equals 还要重写 hashCode

为什么重写了 equals 还要重写 hashCode

时间:2023-02-04 22:11:17浏览次数:59  
标签:name equals hashCode Person 重写 public

目录

借鉴:https://blog.csdn.net/weixin_44061521/article/details/128194172

日期:2023年2月4日21:52:49

如有不足或不对的地方,欢迎指正错误。

先说结论:在 java 中,当往哈希(散列)集合中添加元素时,会先去判断 hashCode 值是否相同,如果不同,则直接插入。如果相同,才去用 equals 判断

注意:

  • 两个对象的 hashCode 值如果不同,则对象一定不同。

  • 两个对象的 hashCode 值如果相同,则对象可能相同,也可能不同,需要用 equals 去比较。

先用 hashCode 去判断对象是否相同可以提升效率,避免直接用 euqals 判断节省了时间。

这也就是为什么用了 hashCode 之后还要用 equals 方法去判断。

下面展开说明一下:

1. 先说一下为什么要重写 equals 方法?

对于引用类型来说,如果不重写 equals 方法,即使调用 equals 方法,默认还是是通过 == 来比较两个对象的地址。

没有重写 equals 方法,Person 类会调用其父类 Object 的 equals 方法。

  • Object 类的 equals 方法
public boolean equals(Object obj) {
    return (this == obj);
}

但在实际业务中,我们都是通过 属性是否相等 来判断两个对象是否为同一个。

创建一个 Person 类用来测试,属性只有一个 name,方便测试:

  • Person 类
public class Person {
   private String name;

   public Person() {
   }

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

}
  • 测试
public class MyTest {
	public static void main(String[] args) {
		Person person1 = new Person("张三");
		Person person2 = new Person("张三");
		System.out.println(person1.equals(person2)); // false
	}
}

Person 类中重写了 equals 方法之后,通过 name 属性来比较两个对象是否为同一个。

使用 idea 快捷键 Ctrl + insert 生成。

@Override
public boolean equals(Object o) {
   if (this == o) return true;
   if (o == null || getClass() != o.getClass()) return false;
   Person person = (Person) o;
   return Objects.equals(name, person.name);
}

测试:

public class MyTest {
   public static void main(String[] args) {
      Person person1 = new Person("张三");
      Person person2 = new Person("张三");
      System.out.println(person1.equals(person2)); // true
   }
}

以上就是为什么重写 equals 方法。

2. 再说为什么还要重写 hashCode 方法?

如果没有重写 hashCode 方法,当我们将对象存储在哈希(散列)集合中会出问题

先在 Person 类中重写 toString 方法方便输出对象内容

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

Set 集合中存储三个同名的对象,输出查看集合的大小以及存储对象的内容

public class MyTest {
   public static void main(String[] args) {
      Person person1 = new Person("张三");
      Person person2 = new Person("张三");
      Person person3 = new Person("张三");

      Set<Person> personSet = new HashSet<>();
      personSet.add(person1);
      personSet.add(person2);
      personSet.add(person3);

      System.out.println(personSet.size());
      personSet.forEach(System.out::println);

   }
}

测试结果:

3
Person{name='张三'}
Person{name='张三'}
Person{name='张三'}

期待的应该存储是一个对象,但却存储了三个,HashSet 没有去重。

这就是没有重写 hashCode 方法造成的。

重写了 hashCode 方法之后

@Override
public int hashCode() {
   return Objects.hash(name);
}

测试结果:

1
Person{name='张三'}

这样就如我们所期待的那样,向集合中存储多个相同的对象,只保留一个。

我们可以在 personSet.add(person1); 处打上断点,Debug 点击下一步就可查看执行步骤了。

或者

  1. 可以将 hashCode 写死返回 1,equals 写死返回 false,再测试可以发现集合中存储了四个对象
  2. 可以将 hashCode 写死返回 1,equals 写死返回 true,再测试可以发现集合中只存储了一个对象

标签:name,equals,hashCode,Person,重写,public
From: https://www.cnblogs.com/sunzhongjie/p/17092522.html

相关文章

  • Java—方法重写
    一、重写与重载的区别重载:在同一个类中,若方法名相同,参数不同(或数据类型不同或个数不同),与返回值与访问修饰符无关;重写:在继承中,方法名相同,参数相同,与返回值与访问......
  • Java方法重写
    如果子类中具有与父类中声明相同的方法,在java中称为方法覆盖。换句话说,如果子类提供了由其父类提供的其中一个方法的特定实现,则它被称为方法覆盖。所以方法覆盖有两个前提......
  • Java 继承中方法的重写
    Java继承中方法的重写关于static的问题解释在构造器中有无static影响着构造器的如图中因为右一和右二中是动态写法所以在main中​因为静态方法是类的方法,而非静态......
  • 自定义 CircleView - 继承 View 重写 onDraw
    一、画一个圆形的View如图,该圆形控件的宽为match_parent,高150dp,为了看到控件的整体宽高效果,为控件加了背景色即浅绿色:#3300aa00该页面的布局<?xmlversion="1.0"encodi......
  • nginx 重写整个 url,结合使用 proxy_pass 和 rewrite
    首先讲下需求背景需要将相同url不同参数的地址转发到服务器上不同的地址举例:example1.com/api.php?act=order->example2.com/api/pay/orderexample1.com/api.php?ac......
  • java (11)方法的重写和super
    1、重写:在Java和其他一些高级面向对象的编程语言中,子类可继承父类中的方法,而不需要重新编写相同的方法。但有时子类并不想原封不动地继承父类的方法,而是想作一定的修改,这......
  • 使用响应头重写功能,解决跨域问题
    借助:ReplaceGoogleCDN响应头重写功能,解决跨域问题缘由:临时起意,想看deno在【微博站点或者微信公众号】中被搜索引擎收录情况,并且把结果显示在指定站点关键点:解决不同站......
  • java map重写 转大写、转驼峰
    /***@authoradmin*@Description*转大写*/publicclassHashMapUpper<V>extendsHashMap<String,V>{/***复写put方法,因为Mybatis底层给Map赋......
  • .htaccess里如何配置www域名与不带www域名301重定向跳转重写URL?
    Apache下web环境,通过修改.htaccess文件即可实现重写URL,很多时候我们需要将带www的域名统一301重定向到不带www的域名,或者将不带www的域名统一301重定向到带www的域名。具体......
  • equals
    Stringa="abc";StringBuilderstringBuilder=newStringBuilder("abc");//判断System.out.println(a.equals(stringBuilder));//System.out.println(stringB......