当前位置: 首页 > news >正文

JAVA学习----HashSet类

HashSet类

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

private transient HashMap<E, Object> map;
private static final Object PRESENT=new Object();

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

Set<Person> set=new HashSet<>();
Person p1=new Person();
Person p2=new Person();
set.add(p1);
set.add(p2);
System.out.println(set.size());

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

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

在Person类中添加方法

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

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

问题在于hashcode值
 

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

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

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

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

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

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

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

散列算法

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

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

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

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

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

public int hashCode() {
        int h = hash;
        if (h == 0 && value.length > 0) {
            hash = h = isLatin1() ? StringLatin1.hashCode(value)
                                  : StringUTF16.hashCode(value);
        }
        return h;
    }

由于自定义类都会直接或者间接的继承于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方法没有任何必然联系,所以说前面的规则是个潜规则

相关文章:

  • 文章组合生成-免费文章组合生成软件
  • 华为面试应该怎么准备?
  • JDBC如何记忆
  • C语言之预处理
  • Flutter 系列---入门篇
  • 全球与中国汽车多楔带行业发展趋向分析及投资前景预测报告2022-2028年
  • Java IO流详解
  • 迷宫_Sarsa算法_边做边学深度强化学习:PyTorch程序设计实践(2)
  • iOS视频捕获入门篇
  • 数据可视化系列教程|六大组件基础知识
  • 《MySQL技术内幕:InnoDB存储引擎》学习笔记-第二章
  • 无线能量传输(WPT)中收发线圈的等效电路模型
  • 回归预测 | MATLAB实现BP神经网络多输入单输出回归预测
  • 新手入门:APP软件测试需要考虑哪些方面
  • .NET序列化 serializable,反序列化
  • [分享]iOS开发 - 实现UITableView Plain SectionView和table不停留一起滑动
  • [译]前端离线指南(上)
  • 《用数据讲故事》作者Cole N. Knaflic:消除一切无效的图表
  • Android交互
  • Docker 笔记(1):介绍、镜像、容器及其基本操作
  • Go 语言编译器的 //go: 详解
  • java中具有继承关系的类及其对象初始化顺序
  • Meteor的表单提交:Form
  • Rancher如何对接Ceph-RBD块存储
  • react-core-image-upload 一款轻量级图片上传裁剪插件
  • Ruby 2.x 源代码分析:扩展 概述
  • thinkphp5.1 easywechat4 微信第三方开放平台
  • 从0搭建SpringBoot的HelloWorld -- Java版本
  • 给第三方使用接口的 URL 签名实现
  • 欢迎参加第二届中国游戏开发者大会
  • 基于Javascript, Springboot的管理系统报表查询页面代码设计
  • 移动互联网+智能运营体系搭建=你家有金矿啊!
  • Linux权限管理(week1_day5)--技术流ken
  • 阿里云ACE认证学习知识点梳理
  • 好程序员大数据教程Hadoop全分布安装(非HA)
  • 通过调用文摘列表API获取文摘
  • ​io --- 处理流的核心工具​
  • (ros//EnvironmentVariables)ros环境变量
  • (八)c52学习之旅-中断实验
  • (八)五种元启发算法(DBO、LO、SWO、COA、LSO、KOA、GRO)求解无人机路径规划MATLAB
  • (附源码)计算机毕业设计高校学生选课系统
  • (转)es进行聚合操作时提示Fielddata is disabled on text fields by default
  • (最优化理论与方法)第二章最优化所需基础知识-第三节:重要凸集举例
  • *(长期更新)软考网络工程师学习笔记——Section 22 无线局域网
  • ***汇编语言 实验16 编写包含多个功能子程序的中断例程
  • ./configure,make,make install的作用
  • .bashrc在哪里,alias妙用
  • .NET Framework 服务实现监控可观测性最佳实践
  • .NET 命令行参数包含应用程序路径吗?
  • .netcore 如何获取系统中所有session_ASP.NET Core如何解决分布式Session一致性问题
  • .Net开发笔记(二十)创建一个需要授权的第三方组件
  • .Net下的签名与混淆
  • /proc/stat文件详解(翻译)
  • @Autowired和@Resource装配
  • @autowired注解作用_Spring Boot进阶教程——注解大全(建议收藏!)