HashSet的分析

一、  HashSet概述:

HashSet实现Set接口,由哈希表(实际上是一个HashMap实例)支持。它不保证set 的迭代顺序;特别是它不保证该顺序恒久不变。此类允许使用null元素。

二、  HashSet的实现:

对于HashSet而言,它是基于HashMap实现的,HashSet底层使用HashMap来保存所有元素,因此HashSet 的实现比较简单,相关HashSet的操作,基本上都是直接调用底层HashMap的相关方法来完成,

HashSet的源代码如下:

1 public class HashSet<E>
2    extends AbstractSet<E>
3    implements Set<E>, Cloneable, java.io.Serializable
4{
5    static final long serialVersionUID = -5024744406713321676L;
6
7    // 底层使用HashMap来保存HashSet中所有元素。
8    private transient HashMap<E,Object> map;
9
10    // 定义一个虚拟的Object对象作为HashMap的value,将此对象定义为static final。
11    private static final Object PRESENT = new Object();
12
13    /**
14     * 默认的无参构造器,构造一个空的HashSet。
15     *
16     * 实际底层会初始化一个空的HashMap,并使用默认初始容量为16和加载因子0.75。
17     */
18    public HashSet() {
19    map = new HashMap<E,Object>();
20    }
21
22    /**
23     * 构造一个包含指定collection中的元素的新set。
24     *
25     * 实际底层使用默认的加载因子0.75和足以包含指定
26     * collection中所有元素的初始容量来创建一个HashMap。
27     * @param c 其中的元素将存放在此set中的collection。
28     */
29    public HashSet(Collection<? extends E> c) {
30    map = new HashMap<E,Object>(Math.max((int) (c.size()/.75f) + 1, 16));
31    addAll(c);
32    }
33
34    /**
35     * 以指定的initialCapacity和loadFactor构造一个空的HashSet。
36     *
37     * 实际底层以相应的参数构造一个空的HashMap。
38     * @param initialCapacity 初始容量。
39     * @param loadFactor 加载因子。
40     */
41    public HashSet(int initialCapacity, float loadFactor) {
42    map = new HashMap<E,Object>(initialCapacity, loadFactor);
43    }
44
45    /**
46     * 以指定的initialCapacity构造一个空的HashSet。
47     *
48     * 实际底层以相应的参数及加载因子loadFactor为0.75构造一个空的HashMap。
49     * @param initialCapacity 初始容量。
50     */
51    public HashSet(int initialCapacity) {
52    map = new HashMap<E,Object>(initialCapacity);
53    }
54
55    /**
56     * 以指定的initialCapacity和loadFactor构造一个新的空链接哈希集合。
57     * 此构造函数为包访问权限,不对外公开,实际只是是对LinkedHashSet的支持。
58     *
59     * 实际底层会以指定的参数构造一个空LinkedHashMap实例来实现。
60     * @param initialCapacity 初始容量。
61     * @param loadFactor 加载因子。
62     * @param dummy 标记。
63     */
64    HashSet(int initialCapacity, float loadFactor, boolean dummy) {
65    map = new LinkedHashMap<E,Object>(initialCapacity, loadFactor);
66    }
67
68    /**
69     * 返回对此set中元素进行迭代的迭代器。返回元素的顺序并不是特定的。
70     *
71     * 底层实际调用底层HashMap的keySet来返回所有的key。
72     * 可见HashSet中的元素,只是存放在了底层HashMap的key上,
73     * value使用一个static final的Object对象标识。
74     * @return 对此set中元素进行迭代的Iterator。
75     */
76    public Iterator<E> iterator() {
77    return map.keySet().iterator();
78    }
79
80    /**
81     * 返回此set中的元素的数量(set的容量)。
82     *
83     * 底层实际调用HashMap的size()方法返回Entry的数量,就得到该Set中元素的个数。
84     * @return 此set中的元素的数量(set的容量)。
85     */
86    public int size() {
87    return map.size();
88    }
89
90    /**
91     * 如果此set不包含任何元素,则返回true。
92     *
93     * 底层实际调用HashMap的isEmpty()判断该HashSet是否为空。
94     * @return 如果此set不包含任何元素,则返回true。
95     */
96    public boolean isEmpty() {
97    return map.isEmpty();
98    }
99
100    /**
101     * 如果此set包含指定元素,则返回true。
102     * 更确切地讲,当且仅当此set包含一个满足(o==null ? e==null : o.equals(e))
103     * 的e元素时,返回true。
104     *
105     * 底层实际调用HashMap的containsKey判断是否包含指定key。
106     * @param o 在此set中的存在已得到测试的元素。
107     * @return 如果此set包含指定元素,则返回true。
108     */
109    public boolean contains(Object o) {
110    return map.containsKey(o);
111    }
112
113    /**
114     * 如果此set中尚未包含指定元素,则添加指定元素。
115     * 更确切地讲,如果此 set 没有包含满足(e==null ? e2==null : e.equals(e2))
116     * 的元素e2,则向此set 添加指定的元素e。
117     * 如果此set已包含该元素,则该调用不更改set并返回false。
118     *
119     * 底层实际将将该元素作为key放入HashMap。
120     * 由于HashMap的put()方法添加key-value对时,当新放入HashMap的Entry中key
121     * 与集合中原有Entry的key相同(hashCode()返回值相等,通过equals比较也返回true),
122     * 新添加的Entry的value会将覆盖原来Entry的value,但key不会有任何改变,
123     * 因此如果向HashSet中添加一个已经存在的元素时,新添加的集合元素将不会被放入HashMap中,
124     * 原来的元素也不会有任何改变,这也就满足了Set中元素不重复的特性。
125     * @param e 将添加到此set中的元素。
126     * @return 如果此set尚未包含指定元素,则返回true。
127     */
128    public boolean add(E e) {
129    return map.put(e, PRESENT)==null;
130    }
131
132    /**
133     * 如果指定元素存在于此set中,则将其移除。
134     * 更确切地讲,如果此set包含一个满足(o==null ? e==null : o.equals(e))的元素e,
135     * 则将其移除。如果此set已包含该元素,则返回true
136     * (或者:如果此set因调用而发生更改,则返回true)。(一旦调用返回,则此set不再包含该元素)。
137     *
138     * 底层实际调用HashMap的remove方法删除指定Entry。
139     * @param o 如果存在于此set中则需要将其移除的对象。
140     * @return 如果set包含指定元素,则返回true。
141     */
142    public boolean remove(Object o) {
143    return map.remove(o)==PRESENT;
144    }
145
146    /**
147     * 从此set中移除所有元素。此调用返回后,该set将为空。
148     *
149     * 底层实际调用HashMap的clear方法清空Entry中所有元素。
150     */
151    public void clear() {
152    map.clear();
153    }
154
155    /**
156     * 返回此HashSet实例的浅表副本:并没有复制这些元素本身。
157     *
158     * 底层实际调用HashMap的clone()方法,获取HashMap的浅表副本,并设置到  HashSet中。
159     */
160    public Object clone() {
161        try {
162            HashSet<E> newSet = (HashSet<E>) super.clone();
163            newSet.map = (HashMap<E, Object>) map.clone();
164            return newSet;
165        } catch (CloneNotSupportedException e) {
166            throw new InternalError();
167        }
168    }
} 

三、 相关说明:

对于HashSet中保存的对象,请注意正确重写其equals和hashCode方法,以保证放入的对象的唯一性。

时间: 2024-11-06 10:51:27

HashSet的分析的相关文章

HashMap、HashSet源代码分析其 Hash 存储机制

集合和引用 就像引用类型的数组一样,当我们把 Java 对象放入数组之时,并不是真正的把 Java 对象放入数组中,只是把对象的引用放入数组中,每个数组元素都是一个引用变量. 实际上,HashSet 和 HashMap 之间有很多相似之处,对于 HashSet 而言,系统采用 Hash 算法决定集合元素的存储位置,这样可以保证能快速存.取集合元素:对于 HashMap 而言,系统 key-value 当成一个整体进行处理,系统总是根据 Hash 算法来计算 key-value 的存储位置,这样可

jdk集合常用方法分析之HashSet和TreeSet

HashSet常用方法介绍 public boolean add(E e) public boolean isEmpty() void clear() public Iterator<E> iterator() int size() 详细方法详见JDK帮助文档 Object的equals方法和hashCode方法 1. Object类equals方法的特点: a) 自反性:x.equals(x)应该返回true b) 对称性:x.equels(y)为true,那么y.equals(x)也为tr

计算机程序的思维逻辑 (41) - 剖析HashSet

上节介绍了HashMap,提到了Set接口,Map接口的两个方法keySet和entrySet返回的都是Set,本节,我们来看Set接口的一个重要实现类HashSet. 与HashMap类似,字面上看,HashSet由两个单词组成,Hash和Set,Set表示接口,实现Set接口也有多种方式,各有特点,HashSet实现的方式利用了Hash. 下面,我们先来看HashSet的用法,然后看实现原理,最后我们总结分析下HashSet的特点. 用法 Set接口 Set表示的是没有重复元素.且不保证顺序

关于集合框架的基本的介绍(JDK7)

集合框架中的接口以及继承关系 关于集合框架中的接口,注意下面列出的全都是接口: 接口与具体的实现类蓝色表示接口橙色表示具体实现类(仅仅列出比较常用的一些): 这个图是一个具体的集合框架的接口以及实现类的图,可以和上面的全部接口相对应: 因为collection有很多的子类,为了操作方便,并没有哪个类直接实现了collecion接口,而是直接对collection接口进行继承,仅仅是提供了更加具体的子接口. 关于List ArrayList的基本使用: package com.javase.col

JAVASE总结笔记

代码下载地址:http://download.csdn.net/download/sinat_25926481/9567455 day1 1:面向对象(理解) (1)面向对象是相对应面向过程而言的一种编程思想,是基于面向过程的. (2)思想特点: A:是一种更符合我们思考习惯的思想. B:把复杂的事情简单化. C:让我们从执行着变成了指挥者. (3)案例: A:买电脑 B:吃饭,洗衣服 C:我要喝水 (4)把大象装进冰箱,让大家对面向对象有一点点的了解. 面向过程: class Demo { p

645. Set Mismatch挑出不匹配的元素和应该真正存在的元素

[抄题]: he set S originally contains numbers from 1 to n. But unfortunately, due to the data error, one of the numbers in the set got duplicated to another number in the set, which results in repetition of one number and loss of another number. Given a

五合一平台源码搭建下载

五合一平台源码搭建下载hubawl.com/thread-234-1-1.html 迭代器模式 迭代器模式(Iterator Pattern) 又称游标(Cursor) 模式,是行为型设计模式之一.迭代器模式源于对容器的访问,比如 Java 中的 List.Map.数组等,我们知道对容器对象的访问必然会涉及到遍历算法,我们可以将遍历的方法封装在容器中,或者不提供便利方法.如果我们将便利方法封装在容器中,那么对于容器来说承担了过多的功能,容器类不仅要维护自身内部的数据元素而且还要对外提供遍历的接口

java源码分析之HashSet

http://blog.csdn.net/jzhf2012/article/details/8540696 Java容器类的用途是"保存对象",分为两类:Map--存储"键值对"组成的对象:Collection--存储独立元素.Collection又可以分为List和Set两大块.List保持元素的顺序,而Set不能有重复的元素. 本文分析Set中最常用的HashSet类,并简单介绍和对比LinkedHashSet. 首先对Set接口进行简要的说明. 存入Set的每

java基础----&gt;hashSet的简单分析(一)

对于HashSet而言,它是基于HashMap实现的,底层采用HashMap来保存元素的.今天我们就简单的分析一下它的实现. HashSet的简单分析 一.hashSet的成员变量组成 public class HashSet<E> extends AbstractSet<E> implements Set<E>, Cloneable, java.io.Serializable private transient HashMap<E,Object> map;