Java Set集合通过重写hashCode和equals实现去重

  • 学习借鉴:https://blog.csdn.net/winy_lm/article/details/50946488

  • 需求:在嵌套获取分类节点子节点的时候,需要对获取的字节点进行去重,所以采用了Set集合,也就是没有重复数据的特性

  • 具体解决思路
    在Set中如果存储类型是基本类型,也就是系统已经封装好的类型,那么就不需要重写hashCode和equals这两个方法,但是如果是我们自己定义的Bean类型,就需要对如何判断重复进行自定义。

  • Set去重流程

我在里面看到了很重要的一点:先判断hashCode是否相同再判断equals是否相同

      for (Entry<K,V> e = table[i]; e != null; e = e.next) {//每添加一个,则循环判断是否与map中的元素相等
            Object k;
            // 先判断hashcode是否一致,然后再判断值是否相等
            if (e.hash == hash && ((k = e.key) == key || key.equals(k))) {
                V oldValue = e.value;
                e.value = value;
                e.recordAccess(this);
                return oldValue;
            }
        }

  • 复杂对象解决思路


/**
 * 类描述:set集合针对String 类型和8大基础数据类型  过滤掉重复数据,
 * 如果存放的是其他类型对象,则需要重写hashCode方法和equals方法,
 * 当hashcode相等时(先执行hashCode方法),则会去执行equals方法,比较每个属性的值
 * 如果一致的话,则不会存进set,否则加入set集合 
 * 
public class MmallCategory implements Serializable {
    private Integer id;
    private Integer parentId;
    private static final long serialVersionUID = 1L;
    public Integer getId() {
        return id;
    }
    public void setId(Integer id) {
        this.id = id;
    }
    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        MmallCategory category = (MmallCategory) o;
        //只是比较id,只有属性值一致才会返回true
        return id != null ? id.equals(category.id) : category.id == null;
    }

    /**
     * 重写hashCode方法,返回的hashCode不一样才会再去比较每一个属性的值
     * @return
     */
    @Override
    public int hashCode() {
        //return id.hashCode
        return id != null ? id.hashCode() : 0;
    }

总结:根据实现类和Set流程的关键代码我就明白了。

根据Set流程博客的链接,HashSet的add方法其实依赖于HashMap的put方法

public boolean add(E e) {
	return map.put(e, PRESENT)==null;
    }

然后我们继续看HashMap put方法的具体实现

public V put(K key, V value) {
        if (key == null)
            return putForNullKey(value);
            //获取hashCode
        int hash = hash(key.hashCode());
        //这个应该似乎是找出对应index索引
        int i = indexFor(hash, table.length);
        //每添加一个,则循环判断是否与map中的元素相等
        for (Entry<K,V> e = table[i]; e != null; e = e.next) {
            Object k;
            // 先判断hashcode是否一致,然后再判断值是否相等
            if (e.hash == hash && ((k = e.key) == key || key.equals(k))) {
                V oldValue = e.value;
                e.value = value;
                e.recordAccess(this);
                return oldValue;
            }
        }
 	//遍历完不重复
        modCount++;
        addEntry(hash, key, value, i);
        return null;
    }

简单点说就是对Set中的元素循环遍历,判断条件是hashCode和equals不能重复,或者hashCode相同时,equals不能相同。这样才能判断去重,可以继续添加

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
掌握集合的概念、体系结构、分类及使用场景 2)了解Set接口及主要实现类(HashSet、TreeSet) 3)了解List接口及主要实现类(ArrayList、LinkedList、Vector) 4)了解Map接口及主要实现类(HashMap、TreeMap、HashTable) 二、实验内容及步骤 1、编写程序练习将以下5个Person类的对象放在一个HashSet中。 姓名:张三 身份证号:178880001 姓名:王五 身份证号:178880002 姓名:李四 身份证号:178880003 姓名:王五 身份证号:178880002 姓名:李四 身份证号:178880004 注意:因为Person类是自定义类,需要重写hashCode()方法和equals()方法,并规定只有姓名和身份证号都相等,则对象相等。 其中计算哈希码的算法:(31 + ((name == null) ? 0 : name.hashCode()))*31 + id (注:name:Person对象的姓名,id:Person对象的身份证号) 主方法中作如下测试: 1)创建一个可放置Person类对象的HashSet; 2)依次添加上述5个对象到HashSet中; 3)把集合中的元素打印出来(使用迭代器Iterator) 2、编写程序练习List集合的基本使用: 1) 创建一个只能容纳String对象名为names的ArrayList集合; 2)按顺序往集合中添加5个字符串对象:"张三"、"李四"、"王五"、"马六"、"赵七"; 3)对集合进行遍历,分别打印集合中的每个元素的位置与内容; 4)打印集合的大小,然后删除集合中的第3个元素,并显示删除元素的内容,然后再打印目前集合中第3个元素的内容,并再次打印集合的大小。 3、编写程序练习Map集合的基本使用: 1)创建一个只能容纳String对象的person的HashMap集合; 2)往集合中添加5个"键-值"对象: "id"-"1"; "name"-"张三"; "sex"-"男"; "age"-"25"; "hobby"-"爱学Java" 3)对集合进行遍历,分别打印集合中的每个元素的键与值; 4)打印集合的大小,然后删除集合中的键为age的元素,并显示删除元素的内容,并再次打印集合的大小。 四、思考题 1、集合中的List、Set、Map有哪些不同? 2、为什么使用集合框架,而尽可能少用数组作为存储结构? 3、如何使用TreeSet实现第一题?

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值