• JAVA学习----HashSet类


    HashSet类

    HashSet类直接实现了Set接口, 其底层其实是包装了一个HashMap去实现的
     以需要存储的数据作为map的key值,以常量PRESENT作为value值

    1. private transient HashMap map;
    2. private static final Object PRESENT=new Object();

    HashSet采用HashCode算法来存取集合中的元素,因此具有比较好的读取和查找性能

    1. Set set=new HashSet<>();
    2. Person p1=new Person();
    3. Person p2=new Person();
    4. set.add(p1);
    5. set.add(p2);
    6. System.out.println(set.size());

    这里显示值为2,因为 Person类中的hashCode方法和equals方法都来自于Object类

    1. Set set=new HashSet<>();
    2. Person p1=new Person(1L,"yan");
    3. Person p2=new Person(1L,"yan");
    4. set.add(p1);
    5. set.add(p2);
    6. System.out.println(p1.equals(p2));
    7. System.out.println(set.size()); //返回值仍旧为2

    在Person类中添加方法

    1. public boolean equals(Object obj){
    2. if(obj!=null && obj instanceof Person){
    3. Person p=(Person)obj;
    4. //具体的比较内容取决于业务规则,这里不进行判空了(偷懒)
    5. return this.id==p.id && this.name.equals(p.name);
    6. }
    7. return false;
    8. }

    再次运行程序输出值还是2  ?

    问题在于hashcode值
     

    1. public int hashCode(){
    2. return this.id.hashCode();
    3. }

    再次运行程序输出值还是1,问题解决

    原因在于:向HashSet中添加元素时首先执行的是对象的hashcode值比较,如果两个
    对象的hashcode值相等时才会继续调用equals方法;如果两个对象的hashcode值不
    相等则不会调用equals方法

    潜规则:不是Java的语法强制要求
        要求当两个对象的equals为true时,hashCode值必须相等

    向set中添加元素到底比较是采用==还是equals?

    1. Set set=new HashSet<>();
    2. String s1="abc";
    3. String s2=new String("abc");
    4. System.out.println(s1==s2);
    5. set.add(s1);
    6. set.add(s2);
    7. System.out.println(set.size()); //返回为1

    HashSet实际上是通过使用HashMap的key实现的,所有key对应的value都是一个常量

    散列算法

    散列法Hashing是一种将字符组成的字符串转换为固定长度(一般是更短长度)的数值
    或索引值的方法,称为散列法,也叫哈希法。

    由于通过更短的哈希值比用原始值进行数据库搜索更快,这种方法一般用来在数据库中
    建立索引并进行搜索,同时还用在各种解密算法中

    当然在存储时需要解决哈希碰撞问题

    通常处理碰撞的方法有开放寻址Open Addressing法和链地址法。

    String类型中的hashCode方法的实现:

    1. public int hashCode() {
    2. int h = hash;
    3. if (h == 0 && value.length > 0) {
    4. hash = h = isLatin1() ? StringLatin1.hashCode(value)
    5. : StringUTF16.hashCode(value);
    6. }
    7. return h;
    8. }

    由于自定义类都会直接或者间接的继承于java.lang.Object,所以所有的类中都有hashCode
    方法

    public native int hashCode();

    HashSet的特征

    • 无序:不仅不能保证元素插入的顺序(如果需要顺序则可以使用LinkedHashSet) ,而且在元素在以后的顺序中也可能变化(这是由HashSet按HashCode存储对象(元素)决定的,对象变化则可能导致HashCode变化)如果需要访问的顺序和插入的顺序一致,可以使用HashSet的子类LinkedHashSet
    • 不允许重复 [equals和hashcode]
    • HashSet是线程非安全的,方法上没有同步约束

    如何判断两个对象相等

    • 实现Set接口的HashSet,依靠HashMap来实现的。

    我们应该为要存放到散列表的各个对象定义hashCode()和equals()

    HashSet的equals和hashCode

    Set集合是不允许重复元素的,否则将会引发各种奇怪的问题。

    那么HashSet如何判断元素重复呢?

    HashSet需要同时通过equals和HashCode来判断两个元素是否相等,具体规则是,如果两个元素通过equals为true,并且两个元素的hashCode相等,则这两个元素相等(即重复)。

    所以如果要重写保存在HashSet中的对象的equals方法,也要重写hashCode方法,重写前后hashCode返回的结果相等(即保证保存在同一个位置)。所有参与计算 hashCode() 返回值的关键属性,都应该用于作为 equals() 比较的标准。

    结论:要求当两个同类型对象equals为true时,必须hashCode值一致。事实上equals方法和hashCode方法没有任何必然联系,所以说前面的规则是个潜规则

  • 相关阅读:
    Flutter高仿微信-第45篇-群聊-文本
    Linux内存管理:memblock
    基于LS-SVM对偶问题的分类、回归、时间序列预测和无监督学习研究(Matlab代码实现)
    [工业互联-4]:工业有线互联总线之IO-Link
    设计模式-抽象工厂模式(Abstract Factory Pattern)结构|原理|优缺点|场景|示例
    Cocos Creator 3D资源的导入与动画播放详解
    [附源码]java毕业设计基于的网上饮品店
    蓝桥杯真题:纯质数
    Hbase(一)入门
    springboot整合搭建webservice项目
  • 原文地址:https://blog.csdn.net/weixin_64771061/article/details/126670162