Java中HashSet的解读
一. HashSet源代码
HashSet 的实现
对于 HashSet 而言,它是基于 HashMap 实现的,HashSet 底层采用 HashMap 来保存所有元素,因此 HashSet 的实现比较简单,查看 HashSet 的源代码,可以看到如下代码: Java代码
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(); } ... }
由上面源程序可以看出,HashSet 的实现其实只是封装了一个 HashMap 对象来存储所有的集合元素,所有放入 HashSet 中的集合元素实际上由 HashMap 的 key 来保存,而 HashMap 的 value 则存储了一个 PRESENT,它是一个静态的 Object 对象。
HashSet 的绝大部分方法都是通过调用 HashMap 的方法来实现的,因此 HashSet 和 HashMap 两个集合在实现本质上是相同的。
二.HashSet<E>对象
HashSet<E>创建的对象称为集合:
HashSet<string> s= HashSet<string>;
那么s就是一个可以存储 string 类型数据的集合,s 可以调用add(String s)方法将string类型的数据添加到集合中。添加到集合中的数据称为集合的元素。集合不允许有相同的元素,也就是说,如果b已经是集合中的元素,那么在执行set.add(b)操作是无效的。 三.HashSet<E>中的方法详解
① public boolean add(E o) —— 向集合添加参数指定的元素。(如果此集合中还
不包含指定元素,则添加指定元素。)
参数:
o - 将添加到此集合中的元素。 返回:
如果该集合还不包含指定元素,则返回 true。
② public void clear() ——清空集合,使集合不含有任何元素。
③ public boolean contains(Object o) —— 判断参数指定的数据是否属于集合。()
参数:
o - 其在此集合中的存在已得到测试的元素。
返回:
如果此集合不包含指定元素,则返回 true。
④ public boolean isEmpty()—— 判断集合是否为空。
返回:
如果此集合不包含任何元素,则返回 true。
⑤ public boolean remove(Object o) ——集合删除参数指定的元素。
参数:
o - 如果存在于此集合中则需要将其移除的对象。
返回:
如果集合包含指定元素,则返回 true。
⑥ public int size () —— 返回集合中元素的个数。
返回:
此集合中的元素的数量(集合的容量)。
⑦ Object [ ] toArray() —— 将集合元素存放到数组中,并返回这个数组。
⑧ boolean containsAll (HashSet set) —— 判断当前集合是否包含参数指定的集合。
⑨ public Object clone () —— 得到当前集合的一个克隆对象,该对象中元素的改
变不会影响到当前集合中的元素,反之亦然。 返回:
此集合的浅表复制。
四.代码:
import java.util.HashSet;
class Student{ String name; int score;
Student(String name, int score){ this.name = name; this.score = score; } }
public class TestHashSet{
public static void main(String[] args) {
Student zh = new Student("张三",77); Student li = new Student("李四",68); Student wa = new Student("王五",67);
HashSet<Student> set = new HashSet<Student>(); HashSet<Student> subset = new HashSet<Student>();
System.out.println("Before add(zh):"+set.add(zh)); set.add(zh);
System.out.println("After add(zh):"+set.add(zh)); //add方法的实现 set.add(wa); set.add(li);
subset.add(wa); subset.add(li);
System.out.println("Is subset empty? "+subset.isEmpty()); //isempty方法的实现
if (set.contains(wa)){
System.out.println("集合set中含有:"+wa.name); //contains方法的实现 }
if(set.containsAll(subset)){
System.out.println("集合set包含集合subset"); }
int number = subset.size(); //size方法的实现 System.out.println("集合subset中有"+number+"个元素:");
Object s[] = subset.toArray(); //toArray 方法的实现
for(int i=0; i<s.length; i++){
System.out.printf("姓名:%s,分数:%d\n", ((Student)s[i]).name,((Student)s[i]).score); }
subset.remove(wa); //remove方法的实现 System.out.println("After implement remove on subset");
System.out.println("集合subset中有"+subset.size()+"个元素:");
Object a[] = subset.toArray();
for(int i=0; i<a.length; i++){
System.out.printf("姓名:%s,分数:%d\n", ((Student)a[i]).name,((Student)a[i]).score); } } }
五.运行结果
Before add(zh):true After add(zh):false Is subset empty? false 集合set中含有:王五 集合set包含集合subset 集合subset中有2个元素: 姓名:李四,分数:68 姓名:王五,分数:67
After implement remove on subset 集合subset中有1个元素: 姓名:李四,分数:68
Java中HashSet的解读的更多相关文章
- Java中HashSet和HashMap
Set中存储元素为什么不重复(即使hashCode相同)? HashSet中存放自定义类型元素时候,需要重写对象中的hashCode方法和equals方法, HashSet中存放自定义类型元素时候,需 ...
- Java中HashSet的重复性与判等运算重载
目录 还有一个故事--(平行世界篇) 还有一个美丽的梦幻家园:java.util 并且还有一个善战的达拉崩巴:HashSet 还有另外一个故事(不是虚假传说) 还有一对涂满毒药的夺命双匕:equals ...
- Java中HashSet,HashMap和HashTable的区别
HashMap.HashSet.HashTable之间的区别是Java程序员的一个常见面试题目,在此仅以此博客记录,并深入源代码进行分析: 在分析之前,先将其区别列于下面 1:HashSet底层采用的 ...
- java中HashSet详解(转)
HashSet 的实现 对于 HashSet 而言,它是基于 HashMap 实现的,HashSet 底层采用 HashMap 来保存所有元素,因此 HashSet 的实现比较简单,查看 HashSe ...
- java中HashSet详解
HashSet 的实现 对于 HashSet 而言,它是基于 HashMap 实现的,HashSet 底层采用 HashMap 来保存所有元素,因此 HashSet 的实现比较简单,查看 HashSe ...
- java集合(4)- java中HashSet详解
HashSet 的实现 对于 HashSet 而言,它是基于 HashMap 实现的,HashSet 底层采用 HashMap 来保存所有元素,因此 HashSet 的实现比较简单,查看 HashSe ...
- java中hashSet原理
转自: http://blog.csdn.net/guoweimelon/article/details/50804799 HashSet是JavaMap类型的集合类中最常使用的,本文基于Java1. ...
- java中HashSet实现(转)
hashset底层的数据结构是hash表,hash表实现方式采用数组+链表,数组类型为HashNode,每个数组元素为链表的头指针,链表中存储经过hash函数散列后冲突的元素,数组的长度为26 has ...
- java中HashSet对象内的元素的hashCode值不能变化
因为不管是HashMap(或HashTable,还是HashSet),key值是以hashCode值存进去的,加入key值变了,将无法从集合内删除对象,导致内存溢出.
随机推荐
- LOJ#6045. 「雅礼集训 2017 Day8」价(最小割)
题面 传送门 题解 首先先把所有权值取个相反数来求最大收益,因为最小收益很奇怪 然后建图如下:\(S\to\)药,容量\(\inf+p_i\),药\(\to\)药材,容量\(\inf\),药材\(\t ...
- 3D-2D:PnP
PnP(Perspective-n-Point):当知道n个3D空间点及其投影位置时,估计相机位姿. 2D-2D的对极几何方法需要八个或八个以上的点对(以八点法为例),且存在着初始化.纯旋转和尺度的问 ...
- Substr函数助你免杀php脚本
主要利用substr函数和url编解码 本文作者:i春秋签约作家Laimooc 1]安全狗: 新研究的php脚本木马:最新版安全狗扫描如下: 成功看到:扫描已完成,未发现网页木马以及其他威胁(开心吗, ...
- [实战] SSH 图形化转发
[实战] SSH 图形化转发 一.介绍 Unix Like操作系统不是只能进行服务器的架设而已,在美编.排版.制图.多媒体应用上也是有其需要的.这些需求都需要用到图形介面(Graphical User ...
- json和csv文件存储
一. json 1:基本概念 1.1 Json和Javascript JSON, 全称JavaScript Object Notation,它通过对象和数组的组合来表示数据.在JavaScript中一 ...
- JavaScript创建对象的4种方法
我们有很多种方式去构造一个对象.可以构造一个对象字面量,也可以和new前缀连用去调用一个构造器函数,或者可以使用Object.create方法去构造一个已经存在的对象的新实例,还可以调用任意一个会返回 ...
- 实验二 输出“Hello Word!”,测试主方法的输入参数和心得体会
实验二 一.输出“Hello World!” 1.首先打开eclipse,如下图所示. 2.选择一个工作空间,如下图所示. 3.接下来,点击[Java项目]创建一个Java项目,如下图所示. 4.然后 ...
- leetcode-77-组合
题目描述: 给定两个整数 n 和 k,返回 1 ... n 中所有可能的 k 个数的组合. 示例: 输入: n = 4, k = 2 输出: [ [2,4], [3,4], [2,3], [1,2], ...
- 2016级算法第六次上机-B.ModricWang's FFT : EASY VERSION
1114 ModricWang's FFT EASY VERSION 思路 利用FFT做大整数乘法,实际上是把大整数变成多项式,然后做多项式乘法. 例如,对于\(1234\),改写成\(f(x)=1* ...
- java的MethodHandle类详解
一.总述 java7为间接调用方法提供了MethodHandle类,即方法句柄.可以将其看作是反射的另一种方式. 这是使用MethodHandle调用方法的一个例子: public class T ...