Java集合–HashSet易错题全面分析

导读:本篇文章讲解 Java集合–HashSet易错题全面分析,希望对大家有帮助,欢迎收藏,转发!站点地址:www.bmabk.com

注意Person类按照id和name已经重写了hashCode和equals方法,问下面代码输出什么?

import java.util.HashSet;
import java.util.Objects;

public class Homework06 {
    public static void main(String[] args) {
        HashSet set = new HashSet();
        Person p1 = new Person(1001, "AA");
        Person p2 = new Person(1002, "BB");
        set.add(p1);
        set.add(p2);
        p1.name = "CC";
        set.remove(p1);
        System.out.println(set);
        set.add(new Person(1001, "CC"));
        System.out.println(set);
        set.add(new Person(1001, "AA"));
        System.out.println(set);
    }
}
class Person{
    public int id;
    public String name;

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

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

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

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

输出结果

[Person{id=1002, name='BB'}, Person{id=1001, name='CC'}]
[Person{id=1002, name='BB'}, Person{id=1001, name='CC'}, Person{id=1001, name='CC'}]
[Person{id=1002, name='BB'}, Person{id=1001, name='CC'}, Person{id=1001, name='CC'}, Person{id=1001, name='AA'}]

具体分析

p1和p2是一定可以加入的,直接跳过。
Java集合--HashSet易错题全面分析通过debug,可以看见存放在table表的7和8位置上。
然后执行p1.name = “CC”;,通过debug,可以看见name=’AA’变成了name=‘CC’。
Java集合--HashSet易错题全面分析set.remove(p1);,表示删除Person(1001, “CC”),我们知道虽然table表8的位置是Person(1001, “CC”),但是索引8是通过Person(1001, “AA”)计算出的,所以Person(1001, “CC”)的索引值不一定是8,即使索引是8,hash值也不相同,所以p1不能被删除。
所以第一次输出是[Person{id=1002, name=‘BB’}, Person{id=1001, name=‘CC’}]。
虽然HashCode方法被重写了,但是底层还是计算对象的hash值,根据hash值决定,虽然table表8的位置是Person(1001, “CC”),与加入对象的内容一样,但是索引8是通过Person(1001, “AA”)计算出的,所以虽然这两个对象内容相同,但是终究不是同一个对象,Person(1001, “CC”)可以被加入。
所以第二次输出是[Person{id=1002, name=‘BB’}, Person{id=1001, name=‘CC’}, Person{id=1001, name=‘CC’}]
Java集合--HashSet易错题全面分析凑巧的是,索引相同,但是hash值不同,说明不是同一个对象,所以可以被加入到索引8的链表上。
执行set.add(new Person(1001, “AA”)), Person(1001, “AA”)与p1是同一个对象,所以索引值与hash值一定相同,但是还需要比较equals方法,我们知道equals方法已经被重写了,当id和name相同才算同一个对象,但是 Person(1001, “AA”)与p1的id和name不相同,所以在equals方法已经被重写后,他们被认为不是同一个对象,所以可以被加入到索引8的链表上,当然在加入链表的过程中,还需与链表上的对象逐个比较,此过程本文不再赘述。
Java集合--HashSet易错题全面分析
所以第三次输出是[Person{id=1002, name=‘BB’}, Person{id=1001, name=‘CC’}, Person{id=1001, name=‘CC’}, Person{id=1001, name=‘AA’}]

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/120183.html

(0)
seven_的头像seven_bm

相关推荐

发表回复

登录后才能评论
极客之音——专业性很强的中文编程技术网站,欢迎收藏到浏览器,订阅我们!