大家都说 Java 很简单,的确 Java 入门不难,但是要想深入了解 Java 那不是一朝一夕能够做到的!
学习 Java 最重要的一点是要学习其设计思想和设计理念,比如集合框架、IO框架的设计等。
通过一个实例谈谈 HashSet 与 hashCode、equals 的使用,以及在使用时的注意事项。
设计一个 Person 类,如下:
- package mark.zhang;
-
- public class Person {
-
- private String name;
- private int age;
-
- public Person(String name, int age) {
- super();
- this.name = name;
- this.age = age;
- }
-
- public String getName() {
- return name;
- }
-
- public int getAge() {
- return age;
- }
-
- public void setName(String name) {
- this.name = name;
- }
-
- public void setAge(int age) {
- this.age = age;
- }
-
- @Override
- public String toString() {
- return "age=" + age + ", name=" + name;
- }
-
- }
这个类很简单,两个成员变量以及 set、get 方法,注意这里没有重写 equals、hashCode 方法。为了在打印的时候方便看出结果,重写 toString 方法。
测试类也照样很简单,如下:
- public class TestPerson {
-
- public static void main(String[] args) {
- Set<Person> set = new HashSet<Person>();
- Person p1 = new Person("喜洋洋", 25);
- Person p2 = new Person("懒洋洋", 26);
- Person p3 = new Person("灰太郎", 27);
- set.add(p1);
- set.add(p2);
- set.add(p3);
- System.out.println(set.size() + " 个动画人物!");
-
- for (Person person : set) {
- System.out.println(person);
- }
- }
- }
输出结果,如下所示:
- 3 个动画人物!
- age=27, name=灰太郎
- age=26, name=懒洋洋
- age=25, name=喜洋洋
ok,看懂上面的程序很简单,只要你不是初学 Java 的话!但是今天的主题不是只讨论这段代码的难易程度。
如果在代码中删除一个“人”,很简单,只需要调用 remove 方法即可,如下所示:
这个时候,我需要修改 Person 这个类,重写父类 Object 的两个方法,equals、hashCode,修改之后的代码:
- package mark.zhang;
-
- public class Person {
-
- private String name;
- private int age;
-
- public Person(String name, int age) {
- super();
- this.name = name;
- this.age = age;
- }
-
- public String getName() {
- return name;
- }
-
- public int getAge() {
- return age;
- }
-
- public void setName(String name) {
- this.name = name;
- }
-
- public void setAge(int age) {
- this.age = age;
- }
-
- @Override
- public String toString() {
- return "age=" + age + ", name=" + name;
- }
-
- @Override
- public int hashCode() {
- final int prime = 31;
- int result = 1;
- result = prime * result + age;
- result = prime * result + ((name == null) ? 0 : name.hashCode());
- return result;
- }
-
- @Override
- public boolean equals(Object obj) {
- if (this == obj)
- return true;
- if (obj == null)
- return false;
- if (getClass() != obj.getClass())
- return false;
- Person other = (Person) obj;
- if (age != other.age)
- return false;
- if (name == null) {
- if (other.name != null)
- return false;
- } else if (!name.equals(other.name))
- return false;
- return true;
- }
-
- }
在测试类中,开始这样子做:
- public class TestPerson {
-
- public static void main(String[] args) {
- Set<Person> set = new HashSet<Person>();
- Person p1 = new Person("喜洋洋", 25);
- Person p2 = new Person("懒洋洋", 26);
- Person p3 = new Person("灰太郎", 27);
- set.add(p1);
- set.add(p2);
- set.add(p3);
- System.out.println(set.size() + " 个动画人物!");
-
- set.remove(p2);
- System.out.println("删除之后," + set.size() + " 个动画人物!");
- for (Person person : set) {
- System.out.println(person);
- }
- }
- }
打印结果:
- 3 个动画人物!
- 删除之后,2 个动画人物!
- age=27, name=灰太郎
- age=25, name=喜洋洋
成功删除一个对象,再次修改测试类的代码:
- public class TestPerson {
-
- public static void main(String[] args) {
- Set<Person> set = new HashSet<Person>();
- Person p1 = new Person("喜洋洋", 25);
- Person p2 = new Person("懒洋洋", 26);
- Person p3 = new Person("灰太郎", 27);
- set.add(p1);
- set.add(p2);
- set.add(p3);
- System.out.println(set.size() + " 个动画人物!");
-
- p2.setName("美人鱼");
-
- set.remove(p2);
- System.out.println("删除之后," + set.size() + " 个动画人物!");
- for (Person person : set) {
- System.out.println(person);
- }
- }
- }
打印结果:
- 3 个动画人物!
- 删除之后,3 个动画人物!
- age=26, name=美人鱼
- age=27, name=灰太郎
- age=25, name=喜洋洋
这次怪了,明明删除一个了,怎么还是有三个呢?你会发现,的确删除一个“懒洋洋”,但是“美人鱼”没有被删除!
如果你在 Person 类中,不重写 hashCode 方法,不会有这种现象发生!
这里说明一个问题:添加到集合的类,不要轻易去修改该类对象的属性,否则 remove() 方法无效。同理 contains() 方法也会无效。
如果有兴趣的话,可以看看其源码,可以看出这与 hashCode() 方法有很大关系!
再说一个容易让人误解的问题:
Collection接口的子接口 List 和 Set,Set (包括其子类)无序不可重复,List (包括其子类)有序可重复,所谓有序无序是相对于 add 的程序执行顺序来说的。
换句话说,对于上面的 List、Set 以及其子类等,如果 equals 为 true 的话,就算是重复的对象。这里的 equals 比较的是内容,不是对象地址。
只不过,对于 Set 来说不可以添加重复对象,对于 List 来说可以添加重复对象!
对于添加对象到Set集合中,从源码可以看出其流程是这样子的:
将对象放入到集合中时,首先判断要放入对象的hashcode值与集合中的任意一个元素的hashcode值是否相等,如果不相等直接将该对象放入集合中。
如果hashcode值相等,然后再通过equals方法判断要放入对象与集合中的任意一个对象是否相等,如果equals判断不相等,直接将该元素放入到集合中,否则不放入。
举个例子,注意:Person 要重写 hashCode、equals 方法:
- public static void main(String[] args) {
- LinkedList<Person> list = new LinkedList<Person>();
- Set<Person> set = new HashSet<Person>();
- Person p1 = new Person("喜喜", 3);
- Person p2 = new Person("喜喜", 3);
- System.out.println("stu1 == stu2 : " + (p1 == p2));
- System.out.println("stu1.equals(stu2) : " + p1.equals(p2));
-
- list.add(p1);
- list.add(p2);
- System.out.println("list size:" + list.size());
-
- set.add(p1);
- set.add(p2);
- System.out.println("set size:" + set.size());
- }
打印结果:
- stu1 == stu2 : false
- stu1.equals(stu2) : true
- list size:2
- set size:1
感谢下面两篇博客,我只是在它们的基础之上添枝加叶。
分享到:
相关推荐
在Java编程语言中,`equals()` 和 `hashCode()` 方法是Object类中的两个核心方法,所有类都默认继承自Object类。这两个方法在处理对象比较和集合操作时起着至关重要的作用。当我们创建自定义类并需要对对象进行精确...
Java容器集合(equals和hashCode+基础数据结构+ArrayList+Vector和LinkedList) Java容器集合是Java中的一种基础数据结构,用于存储和管理数据。其中,equals和hashCode方法是Java容器集合中两个非常重要的方法,...
在Java编程语言中,`equals()` 和 `hashCode()` 方法是两个非常重要的成员,尤其是在处理对象比较和集合操作时。这两个方法通常与`Object`类中的默认实现相关联,但为了在实际开发中实现正确的对象比较和哈希表操作...
本文还介绍了定义对象的相等性、实施equals()和hashCode()的需求、编写自己的equals()和hashCode()方法。通过统一定义equals()和hashCode(),可以提升类作为基于散列的集合中的关键字的使用性。
在Java编程语言中,`equals()`和`hashCode()`方法是对象的基本组成部分,它们主要用于对象的比较和存储。这两个方法在`java.lang.Object`类中定义,因此所有的Java类都默认继承了这两个方法。然而,根据具体的应用...
在Java编程语言中,`equals()`和`hashCode()`方法是两个非常重要的概念,尤其是在处理对象比较和容器(如HashMap和HashSet)操作时。这两个方法在Java的类库中有着核心地位,尤其是对于类实例的比较和存储。接下来,...
"关于Object中equals方法和hashCode方法判断的分析" 在 Java 中,Object 类提供了两个重要的方法:equals 方法和 hashCode 方法。这两个方法都是用于比较两个对象是否相等的,但它们的实现机理和作用域却有所不同。...
深入理解equals和hashCode方法 equals和hashCode方法是Java中Object类提供的两个重要方法,对以后的学习有很大的帮助。本文将深入剖析这两个方法,帮助读者更好地理解和使用它们。 equals方法 equals方法是用于...
1.概述 2.为什么重写equels方法要重写hashcode方法 3.例子
在Java编程语言中,`equals()`, `hashCode()` 和 `toString()` 是三个非常重要的方法,它们主要用于对象的比较、哈希存储以及打印对象信息。这三个方法是Java对象的基础特性,对于理解和开发高质量的Java程序至关...
图文并茂吃透面试题,看完这个,吊打面试官,拿高薪offer!
"Java中equals、hashcode和==的区别" Java 中 equals、hashcode 和==的区别是 Java 编程语言中一个经常遇到的问题。这三个概念都是用来比较对象的,但是它们之间存在着本质的区别。 首先,==号是Java中的一个...
Java重写equals同时需要重写hashCode的代码说明,以及如何重写hashCode方法,此代码演示按照effective java书籍说明的重写思路。代码中演示了使用集合存储对象,并且对象作为key,需重写equals和hashCode.
equals 方法和 hashCode 方法是 Java 语言中两个重要的方法,它们都是在 Object 类中定义的。equals 方法用于比较两个对象是否相等,而 hashCode 方法用于返回对象的哈希码。 在 Java 的 Object 类中,equals 方法...
Java中的equals和hashCode方法是两个非常重要的方法,它们都是Object类中的方法。在实际开发中,正确地重写这两个方法对于确保程序的正确性和性能至关重要。下面,我们将详细地介绍Java重写equals及hashCode方法的...
在Scala中,正确实现`equals`和`hashCode`方法对于集合操作至关重要,因为它们影响了对象在Set和Map中的存储和查找。这个处理器可能提供了一种自动化的方式来生成这些方法,避免了手动实现时可能出现的错误和不一致...
Object 类的 equals 和 hashCode 方法的重要性与实现 在 Java 编程语言中,Object 类是所有类的父类,但是在实际开发中,我们往往需要重写 Object 中的 equals 和 hashCode 方法,以便正确地比较对象的逻辑内容,而...
重写equals()、hashcode()和compareTo()方法是非常重要的,它可以使我们的程序更加高效、可靠和灵活。在实际开发中,我们需要根据具体情况来决定是否需要重写这些方法,并遵守相关的规则,以避免出现意想不到的结果...
equals()和hashcode()这两个方法都是从object类中继承过来的。当String 、Math、还有Integer、Double。。。。等这些封装类在使用equals()方法时,已经覆盖了object类的equals()方法.
在Java编程语言中,`equals()` 和 `hashCode()` 方法是Object类中定义的基本方法,所有类都默认继承自Object类,因此每个Java对象都有这两个方法。这两个方法在处理集合类,尤其是Set接口的实现(如HashSet)时起着...