【Java基础知识总结 | 第十篇】HashSet底层实现原理

在这里插入图片描述

文章目录

  • 10.HashSet底层实现原理
    • 10.1HashSet特点
    • 10.2HashSet源码
    • 10.3 add流程
    • 10.4总结

10.HashSet底层实现原理

10.1HashSet特点

  1. 存储对象:HashSet 存储对象采用哈希表的方式,它不允许重复元素,即集合中不会包含相同的元素。当向 HashSet 中添加元素时,会根据元素的 hashCode() 方法和 equals() 方法来判断元素是否重复。
  2. 底层数据结构:HashSet 的底层数据结构是基于 HashMap 实现的,实际上 HashSet 的元素就是作为 HashMap 的 key 存储的。
  3. null 值问题:HashSet 允许存储一个 null 元素。
  4. 线程安全问题:不安全

10.2HashSet源码

public class HashSet<E> extends AbstractSet<E> implements Set<E>, Cloneable, java.io.Serializable{  // 使用 HashMap 的 key 保存 HashSet 中所有元素  private transient HashMap<E,Object> map;   // 定义一个虚拟的 Object 对象作为 HashMap 的 value   private static final Object PRESENT = new Object();// 构造方法,初始化 HashSet,底层会初始化一个 HashMap   public HashSet(){   map = new HashMap<E,Object>();   }   // 以指定的 initialCapacity、loadFactor 创建 HashSet   // 其实就是以相应的参数创建 HashMap   public HashSet(int initialCapacity, float loadFactor){   map = new HashMap<E,Object>(initialCapacity, loadFactor);   }   public HashSet(int initialCapacity){   map = new HashMap<E,Object>(initialCapacity);   }   HashSet(int initialCapacity, float loadFactor, boolean dummy){   map = new LinkedHashMap<E,Object>(initialCapacity   , loadFactor);   }   // 调用 map 的 keySet 来返回所有的 key   public Iterator<E> iterator(){   return map.keySet().iterator();   }   // 调用 HashMap 的 size() 方法返回 Entry 的数量,就得到该 Set 里元素的个数  public int size(){   return map.size();   }  // 调用 HashMap 的 isEmpty() 判断该 HashSet 是否为空,  // 当 HashMap 为空时,对应的 HashSet 也为空  public boolean isEmpty(){   return map.isEmpty();   }   // 调用 HashMap 的 containsKey 判断是否包含指定 key   //HashSet 的所有元素就是通过 HashMap 的 key 来保存的  public boolean contains(Object o){   return map.containsKey(o);   }   // 将指定元素放入 HashSet 中,也就是将该元素作为 key 放入 HashMap   public boolean add(E e){   return map.put(e, PRESENT) == null;   }   // 调用 HashMap 的 remove 方法删除指定 Entry,也就删除了 HashSet 中对应的元素  public boolean remove(Object o){   return map.remove(o)==PRESENT;   }   // 调用 Map 的 clear 方法清空所有 Entry,也就清空了 HashSet 中所有元素  public void clear(){   map.clear();   }} 

10.3 add流程

  1. HashSet底层使用了哈希表来支持的,特点:存储快;
  2. 往Haset添加元素的时候,HashSet会先调用元素的hashCode方法得到元素的哈希值 ,然后通过元素的哈希值经过移位等运算,就可以算出该元素在哈希表中的存储位置
    1. 如果算出的元素存储的位置目前没有任何元素存储,那么该元素可以直接存储在该位置上;
    2. 如果算出的元素的存储位置目前已经存在有其他的元素了,那么还会调用该元素的equals方法
      与该位置的元素再比较一次,如果equals方法返回的是true,那么该位置上的元素视为重复元
      素,不允许添加,如果返回的是false,则允许添加
  3. 例子:该对象的 age 字段与之前已经添加的一个 age 字段为 18 的对象相同,且哈希码也相同,所以 HashSet 认为它们是同一个对象,因此不会将其添加到集合中,返回的结果是 false
class Person {String name;int age;private void Peron() {// TODO Auto-generated method stub}public Person(String name, int age) {super();this.name = name;this.age = age;}@Overridepublic int hashCode() {System.out.println("--------");// TODO Auto-generated method stubreturn this.age;}@Overridepublic boolean equals(Object obj) {System.out.println("---****----");Person p = (Person)obj;return this.age == p.age;}@Overridepublic String toString() {// TODO Auto-generated method stubreturn "{姓名:"+name+"年龄:"+age+"}";}
}
public class Demo2 {public static void main(String[] args) {HashSet set = new HashSet();set.add(new Person("yy",18));set.add(new Person("xx",19));set.add(new Person("zz",20));set.add(new Person("jj",25));System.out.println("添加元素成功了嗎?"+set.add(new Person("zhangsan",18)));System.out.println("集合的元素:"+set);}
}

10.4总结

  1. 存储对象:HashSet 存储对象采用哈希表的方式,它不允许重复元素,即集合中不会包含相同的元素。当向 HashSet 中添加元素时,会根据元素的 hashCode() 方法和 equals() 方法来判断元素是否重复。
  2. 底层数据结构:HashSet 的底层数据结构是基于 HashMap 实现的,实际上 HashSet 的元素就是作为 HashMap 的 key 存储的。
  3. null 值问题:HashSet 允许存储一个 null 元素。
  4. 线程安全问题:不安全
  5. 当我们试图把某个类的对象当成 HashMap的 key,或试图将这个类的对象放入 HashSet 中保存时,重写该类的equals(Object obj)方法和 hashCode() 方法很重要,而且这两个方法的返回值必须保持一致:当该类的两个的 hashCode() 返回值相同时,它们通过 equals() 方法比较也应该返回 true。通常来说,所有参与计算 hashCode() 返回值的关键属性,都应该用于作为 equals() 比较的标准。

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/794775.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

使用Ajax的优点

Ajax的优点主要表现在以下几个方面&#xff1a; 1. 页面无刷新&#xff1a;Ajax可以在不更新整个页面的前提下与服务器通信&#xff0c;维护数据&#xff0c;给用户的体验非常好。 2. 异步通信&#xff1a;Ajax使用异步方式与服务器通信&#xff0c;不需要打断用户的操作&#…

数据挖掘中的PCA和KMeans:Airbnb房源案例研究

目录 一、PCA简介 二、数据集概览 三、数据预处理步骤 四、PCA申请 五、KMeans 聚类 六、PCA成分分析 七、逆变换 八、质心分析 九、结论 十、深入探究 10.1 第 1 步&#xff1a;确定 PCA 组件的最佳数量 10.2 第 2 步&#xff1a;使用 9 个组件重做 PCA 10.3 解释 PCA 加载和特…

【微服务】------核心组件架构选型

1.微服务简介 微服务架构&#xff08;Microservice Architecture&#xff09;是一种架构概念&#xff0c;旨在通过将功能分解到各个离散的服务中以实现对解决方案的解耦&#xff0c;从而降低系统的耦合性&#xff0c;并提供更加灵活的服务支持。 2.微服务技术选型 区域内容…

Kotlin学习日志(一)TextView、Button、Toast的使用(1)

android:layout_width“wrap_content” android:layout_height“wrap_content”/> import kotlinx.android.synthetic.main.activity_main.* 这句话的意思是引进Kotlin的的控件变量自动映射功能&#xff0c;接下来只要是这个activity_main.xml文件中的控件&#xff0c;我…

c++格式化输出

在 C 中&#xff0c;格式化输出通常使用流插入运算符 << 结合输出流对象&#xff08;如 std::cout&#xff09;来完成。C 标准库提供了一系列的控制符和函数来实现各种格式化输出需求。进行格式化输出时&#xff0c;除了控制输出宽度和精度外&#xff0c;还可以使用其他格…

蓝桥杯第十四届C++A组(未完)

【规律题】平方差 题目描述 给定 L, R&#xff0c;问 L ≤ x ≤ R 中有多少个数 x 满足存在整数 y,z 使得 。 输入格式 输入一行包含两个整数 L, R&#xff0c;用一个空格分隔。 输出格式 输出一行包含一个整数满足题目给定条件的 x 的数量。 样例输入 1 5 样例输出 …

OpenTofu路在何方:定量分析Terraform issue数据,洞察用户需求|OpenTofu Day 闪电演讲

数澈软件 Seal 首席架构师李平辉提交的演讲议题“Alias TerraformTofu. Job’s Done, Now What?”入选 KubeCon EU 同场活动 OpenTofu Day&#xff0c;本文为演讲实录。 大家好&#xff0c;我是 Lawrence&#xff0c;是 Seal 的首席架构师。今天将由我为大家带来 Lightening T…

【接口】HTTP(2) |请求方法及状态码

1、HTTP常用请求方法 get&#xff1a;获取资源或指定的数据 请求指定的页面信息&#xff0c;返回实体主体&#xff08;查询&#xff09; post&#xff1a;发送数据给服务器&#xff0c;创建或更新资源 put&#xff1a;创建/替换目标资源 delete&#xff1a;删除资源 get …

后端开发框架Spring Boot快速入门

写在前面 推荐将本文与Spring Boot 相关知识和工具类一文结合起来看&#xff0c;本文为主&#xff0c;上面那篇文章为辅&#xff0c;一起食用&#xff0c;以达到最佳效果&#xff0c;当然&#xff0c;大佬随意。 IDEA创建Spring Boot工程 关于Spring Boot框架项目&#xff0…

第二节课《轻松玩转书生·浦语大模型趣味 Demo》

比较匆忙&#xff0c;假期前仿照第一期课程的内容好像被清空了&#xff0c;重新搭建一次。 https://github.com/InternLM/Tutorial/blob/camp2/helloworld/hello_world.md 按照那老师写好的&#xff0c;一步步复制就好了 浦语灵笔2的大概率是会超出显存&#xff0c;先不测试了…

MySQL-排序与分页

1. 排序 如果没有使用排序操作&#xff0c;默认情况下查询返回的数据是按照添加数据的顺序显示的。 SELECT * FROM employees;1.1 基本使用 1&#xff09;使用 ORDER BY 对查询到的数据进行排序操作。 升序&#xff1a;ASC(ascend)降序&#xff1a;DESC (descend) 练习&am…

2024.4.4-[作业记录]-day09-CSS 布局模型(标准流模型、浮动模型)

个人主页&#xff1a;学习前端的小z 个人专栏&#xff1a;HTML5和CSS3悦读 本专栏旨在分享记录每日学习的前端知识和学习笔记的归纳总结&#xff0c;欢迎大家在评论区交流讨论&#xff01; 文章目录 作业 2024.4.4-学习笔记1 CSS 布局模型1.1 标准流1.2 CSS 浮动1.3 去除塌陷 2…

【零基础学数据结构】顺序表实现书籍存储

目录 书籍存储的实现规划 ​编辑 前置准备&#xff1a; 书籍结构体&#xff1a; 书籍展示的初始化和文件加载 书籍展示的销毁和文件保存 书籍展示的容量检查 书籍展示的尾插实现 书籍展示的书籍增加 书籍展示的书籍打印 书籍删除展示数据 书籍展示修改数据 在指定位置之前…

SpamSieve mac垃圾邮件过滤器 直装激活版

SpamSieve通过强大的垃圾邮件过滤技术&#xff0c;帮助用户有效管理和消除不想要的电子邮件。它能与多种电子邮件客户端无缝集成&#xff0c;如Apple Mail、Microsoft Outlook、Airmail等。 软件下载&#xff1a;SpamSieve mac直装激活版下载 该软件利用先进的算法和机器学习技…

JS代码小知识(个人向)

JS 对象转数组 let obj {0:"a",1:"b",length:2 //加上这个就能转了 }; console.log(Array.from(obj)); // ["a", "b"] 数组的拼接 let a ["a","b"] let b ["c","d"] let c [...a , …

sass中的导入与部分导入

文章目录 sass中的导入与部分导入1. import&#xff1a;传统的导入方式2. use&#xff1a;现代化的模块化导入 sass中的导入与部分导入 在大型前端项目中&#xff0c;CSS代码量往往十分庞大&#xff0c;为了保持其可读性、可维护性以及便于团队协作&#xff0c;模块化开发成为…

每日一题 第七十期 洛谷 [蓝桥杯 2020 省 AB2] 回文日期

[蓝桥杯 2020 省 AB2] 回文日期 题目描述 2020 年春节期间&#xff0c;有一个特殊的日期引起了大家的注意&#xff1a;2020 年 2 月 2 日。因为如果将这个日期按 yyyymmdd 的格式写成一个 8 8 8 位数是 20200202&#xff0c;恰好是一个回文数。我们称这样的日期是回文日期。…

深度学习方法;乳腺癌分类

乳腺癌的类型很多&#xff0c;但大多数常见的是浸润性导管癌、导管原位癌和浸润性小叶癌。浸润性导管癌(IDC)是最常见的乳腺癌类型。这些都是恶性肿瘤的亚型。大约80%的乳腺癌是浸润性导管癌(IDC)&#xff0c;它起源于乳腺的乳管。 浸润性是指癌症已经“侵袭”或扩散到周围的乳…

TR3 - Transformer算法详解

目录 文本输入处理词向量位置向量 编码器 EncoderSelf-Attention多头注意力机制残差连接 解码器 Decoder线性层与Softmax损失函数总结与心得体会 这周来看一下Transformer是怎么将文本转换成向量&#xff0c;然后又输入到模型处理并得到最终的输出的。 文本输入处理 词向量 …

【opencv】教程代码 —ml (主成分分析、支持向量机、非线性支持向量机)

1. introduction_to_pca.cpp 主成分分析 /*** file introduction_to_pca.cpp* brief 这个程序演示了如何使用OpenCV PCA 提取物体的方向* author OpenCV团队*/// 包含OpenCV函数库所需要的头文件 #include "opencv2/core.hpp" #include "opencv2/imgproc.hpp&q…