### 概述
此类实现Set接口,由哈希表支持(实际上是一个 HashMap集合)。HashSet集合不能保证的迭代顺序与元素存储顺序相同 . 哈希表是单向链 .
HashSet集合,采用哈希表结构存储数据,保证元素唯一性的方式依赖于:hashCode()与equals()方法。
### 实例
~~~
HashSet<String> set = new HashSet<String>();
set.add("我");
set.add("爱");
set.add("你");
Iterator<String> it = set.iterator();
while(it.hasNext()){
System.out.println(it.next());
}
~~~
结果
~~~
你
我
爱
~~~
### 数据结构
哈希表底层使用的也是数组机制,数组中也存放对象,而这些对象往数组中存放时的位置比较特殊,当需要把这些对象给数组中存放时,那么会根据这些对象的特有数据结合相应的算法,计算出这个对象在数组中的位置,然后把这个对象存放在数组中。而这样的数组就称为哈希数组,即就是哈希表。
当向哈希表中存放元素时,需要根据元素的特有数据结合相应的算法,这个算法其实就是Object类中的hashCode方法。由于任何对象都是Object类的子类,所以任何对象有拥有这个方法。即就是在给哈希表中存放对象时,会调用对象的hashCode方法,算出对象在表中的存放位置,这里需要注意,如果两个对象hashCode方法算出结果一样,这样现象称为哈希冲突,这时会调用对象的equals方法,比较这两个对象是不是同一个对象,如果equals方法返回的是true,那么就不会把第二个对象存放在哈希表中,如果返回的是false,就会把这个值存放在哈希表中。
总结:保证HashSet集合元素的唯一,其实就是根据对象的hashCode和equals方法来决定的。如果我们往集合中存放自定义的对象,那么保证其唯一,就必须复写hashCode和equals方法建立属于当前对象的比较方式。
![](https://box.kancloud.cn/7bc010e5d54296f10d7cdfbc7d8f7794_1260x590.png)
### 说明
它是线程不安全的 . HashSet初始容量是16 . 加载因子是0.75 . 如果容量超过16 ,name就会生成新的数组,并和原有的HashSet进行复制 . 这样就会降低性能 .
### 字符串对象的哈希值
对象的哈希值,是一个十进制的整数 . 是父类Object去完成的 ,是一个本地方法 . 如果我们不重写父类方法,每次运行**结果是不同的十进制整数值** . 哈希值是存储到HashSet集合的依据 .
~~~
String s1 = new String("abc");
String s2 = new String("abc");
System.out.println(s1.hashCode());
System.out.println(s2.hashCode()); //哈希值是一致的
~~~
两个对象的内存地址是不一致的,但是哈希值是一致的,为什么呢 ? 因为String类重写了父类的方法, 它有自己计算哈希值的方式 .
### 哈希表的存储过程
~~~
HashSet<String> set = new HashSet<String>();
set.add(new String("hello world"));
set.add(new String("hello world"));
set.add(new String("happy"));
set.add(new String("happy"));
System.out.println(set.toString());
~~~
结果
~~~
[happy, hello world]
~~~
1. 调用每个存储对象的哈希值 .
2. 集合在容器内寻找有没有相同的哈希值对象 .
3. 如果找到有相同哈希值的对象, 将后放入集合的的对象调用equals()方法和第一个进行比较 .
4. 如果两个对象,哈希值一致,equals()方法返回值是true , 集合判断元素已经重复 .
5. 对于判定元素不重复的存储方式是让前一个对象记住后一个对象的内存地址, 和桶状类似 .
6. 简单说HashSet只能存放不同元素的机制就是先比较哈希值,再比较元素内容是否相等 .
### 哈希表存储自定义对象
- 环境
- Java基础
- 数据类型
- 基本类型
- 引用类型
- Array
- Class
- ArrayList
- 流程控制
- ASCII码表
- 面向对象
- 概述
- 变量
- this
- 修饰符
- 封装
- 继承
- final
- static
- 接口
- 接口
- 抽象类
- overload
- 接口和抽象类区别
- 案例
- 多态
- 构造方法
- super
- 匿名对象
- 内部类
- 静态导入
- 包的声明和访问
- 代码块
- 代理设计模式
- 静态代码块
- Object类
- lambda表达式
- 枚举
- 定义枚举
- 常用方法
- 枚举常量
- 异常
- 可变参数
- Properties
- 概述
- Properties和持久化有关的方法
- 反射
- 类加载器
- 概述
- Class类
- 获取构造方法并运行
- 获取成员变量并改值
- 获取成员方法并运行
- 反射泛型擦除
- 反射通过配置文件运行
- 常用 API
- Date类
- DateFormat类
- Calendar类
- Math类
- String类
- object类
- System类
- 正则表达式
- BigInteger类
- BigDecimal类
- Arrays工具类
- StringBuffer类
- 基本类型包装类
- 自动装箱和自动拆箱
- 框架集合
- collection接口
- 概述
- 接口方法
- 迭代器
- 迭代器并发异常
- 增强for循环
- 泛型
- hashCode和equals
- List接口
- 概述
- ArrayList
- LinkedList
- vector
- Set接口
- 概述
- 对象的哈希值
- HashSet
- LinkedHashSet
- 保证自定义元素的唯一性(重写)
- ArrayList和HashSet的contains原理
- Map接口
- 概述
- Map集合常用方法
- 迭代Map
- Map存储自定义类型
- HashMap
- LinkedHashMap
- Hashtable
- 嵌套Map遍历
- 方法的可变参数
- collections工具类
- 集合嵌套
- 集合特点总结
- Collection集合工具类
- Arrays工具类
- File类
- 概述
- 获取方法
- 创建和删除
- 判断方法
- 其他方法
- 文件过滤器
- IO流
- 字节流
- 概述
- 字节流
- 字节流复制文件
- 字节缓冲流
- 复制单级文件夹
- 使用字节流读取中文的乱码问题
- 字符编码表
- 方便程序员的IO流
- 转换流
- 概述
- OutputStreamWriter类
- InputStreamWriter类
- 转换流和其他子类的区别
- 序列化流
- 概述
- 序列化接口以及版本号问题
- transient
- 打印流
- 多线程
- 概述
- main线程介绍
- 开启线程方式一:Thread类
- 开启线程方式二:Runnable类
- 线程的执行原理
- 使用哪种方式好
- 使用匿名内部类创建线程
- 线程安全
- ThreadLocal
- 网络编程
- XML
- 概述
- 语法
- 元素标签
- 属性
- 转义字符
- CDATA区
- XML约束
- DTD约束
- 三种引入DTD的方式
- DTD语法
- Schema约束
- JDBC
- 概述
- JDBC原理
- JDBC查询操作
- JDBCUtils
- 预处理对象
- JDBC连接池
- 概述
- DBCP
- C3P0
- DBUtils操作
- ArrayHandler
- ArrayListHandler
- BeanHandler
- BeanListHandler
- ColumnListHandler
- MapHandler
- MapListHandler
- ScalarHandler
- 事务
- 事务概述
- 原生JDBC完成转账
- DBUtils完成转账
- 事务特性
- 三层思想
- 概述
- JavaWEB
- tomcat
- servlet
- 概述
- 映射servlet
- Servlet接口
- HttpServlet类
- load-on-startup
- servletContext
- ServletRequest
- ServletResponse
- Cooike&Session
- servlet三大域对象
- JSP
- 概述
- jsp内置对象
- 指令
- page
- include
- taglib
- JSP中的四个域对象
- jsp的一个动作标签
- EL表达式
- 概述
- 获取域中数据
- 执行运算
- el表达式中的内置对象
- jsti标签
- 概念
- jstl分类
- Filter过滤器
- 概述
- 生命周期
- url-pattern配置
- filter的chain过滤链
- filter-mapping子标签
- 全局统一错误页面
- filterConfig
- 自动登录实例
- 全局统一编码
- Listener监听器
- 概念
- 监听器分类
- 监听三个对象的创建和销毁
- 监听三个对象属性的变化
- 监听javabean在session中的状态变化
- 定时器
- 概念
- calendar日历
- 邮件
- 概述
- JSON
- 数据结构
- 堆栈结构
- 队列结构
- 数组结构
- 链表结构
- 基础加强
- 面试题:增强一个对象的方法有几种方式?
- 装饰者增强
- 动态代理
- 动态代理
- 动态代理细节
- 使用动态代理解决字符集编码
- 注解
- 概述
- 扩展的自定义注解
- 类加载器
- 全盘负责托管机制
- 开发模式
- 问题
- lambda
- 函数式编程思想概述
- String字符串