jdk1.8.0_45源码解读——HashSet的实现

jdk1.8.0_45源码解读——HashSet的实现

一、HashSet概述

  HashSet实现Set接口,由哈希表(实际上是一个HashMap实例)支持。主要具有以下的特点:

  • 不保证set的迭代顺序,特别是它不保证该顺序恒久不变
  • 有且只允许一个null元素
  • 不允许有重复元素,这是因为HashSet是基于HashMap实现的,HashSet中的元素都存放在HashMap的key上面,而value中的值都是统一的一个private static final Object PRESENT = new Object();
  • 非同步的。如果多 个线程同时访问一个哈希set,而其中至少一个线程修改了该 set,那么它必须保持外部同步。这通常是通过对自然封装该set的对象执行同步操作来完成的。如果不存在这样的对象,则应该使用 Collections.synchronizedSet 方法来“包装” set。最好在创建时完成这一操作,以防止对该set进行意外的不同步访问:
Set s = Collections.synchronizedSet(new HashSet(...));
  • HashSet通过iterator()返回的迭代器是fail-fast的

二、HashSet源码解析

相关HashSet的操作,基本上都是直接调用底层HashMap的相关方法来完成。

1. HashSet类结构

//通过HashSet实现的接口可知,其支持所有集合操作,能被克隆,支持序列化
public class HashSet<E>
    extends AbstractSet<E>
    implements Set<E>, Cloneable, java.io.Serializable
{
    static final long serialVersionUID = -5024744406713321676L;

    private transient HashMap<E,Object> map;

    //定义一个"虚拟"的static final Object对象作为HashMap的value
    private static final Object PRESENT = new Object();

    ......
}

HashSet包含了两个重要的成员变量:map, PRESENT。

(01) map是一个HashMap<E, Object>对象,HashSet是由一个HashMap实例支持的。

(02) PRESENT是一个static final Object对象,用来作为HashMap中的value值。

2. 构造函数

HashSet提供了四种方式的构造器,可以构造一个新的空 set,其底层 HashMap实例的默认初始容量是16,加载因子是 0.75,构造一个包含指定collection中的元素的新set,构造一个新的空set,其底层HashMap实例具有指定的初始容量和默认的加载因子(0.75),以及构造一个新的空set,其底层HashMap实例具有指定的初始容量和指定的加载因子。

    //默认的无参构造器,构造一个空的HashSet,实际底层会初始化一个空的HashMap,并使用默认初始容量为16和加载因子0.75。
    public HashSet() {
        map = new HashMap<E, Object>();
    }

    //构造一个包含指定collection中的元素的新set。
    //实际底层使用默认的加载因子0.75和足以包含指定collection中所有元素的初始容量来创建一个HashMap。
    public HashSet(Collection<? extends E> c) {
        map = new HashMap<E, Object>(Math.max((int) (c.size()/.75f) + 1, 16));
        addAll(c);  //AbstractCollection.addAll(Collection<? extends E> c)
    }

    //以指定的初始容量和加载因子构造一个空的HashSet
    public HashSet(int initialCapacity, float loadFactor) {
        map = new HashMap<E, Object>(initialCapacity, loadFactor);
    }

    //以指定的initialCapacity和默认加载因子0.75构造一个空的HashSet
    public HashSet(int initialCapacity) {
        map = new HashMap<E, Object>(initialCapacity);
    }

    /**
     * 以指定的initialCapacity和loadFactor构造一个新的空链接哈希集合
     * 此构造函数为包访问权限,不对外公开,实际只是是对LinkedHashSet的支持
     *
     * @param      initialCapacity   初始容量
     * @param      loadFactor        加载因子
     * @param      dummy             标记,用于与其他的构造函数区分(可忽略)
     * @throws     IllegalArgumentException 如果初始容量小于零或加载因子为非正数
     */
    HashSet(int initialCapacity, float loadFactor, boolean dummy) {
        map = new LinkedHashMap<E, Object>(initialCapacity, loadFactor);
    }

3.查找

HashSet提供了contains(Object o)查看是否包含指定元素的方法,其底层调用的是HashMap.containsKey(Object key)判断是否包含指定key。

    //如果此set包含指定元素,则返回 true
    public boolean contains(Object o) {
        return map.containsKey(o);
    }

4.添加

HashSet提供了add(E e)添加元素的方法,其调用的是底层HashMap中的put(K key, V value)方法,首先判断元素(也就是key)是否存在,如果不存在则插入,如果存在则不插入,这样HashSet中就不存在重复值。

   //如果此set中尚未包含指定元素,则添加指定元素
    public boolean add(E e) {
        return map.put(e, PRESENT)==null;
    }

底层:当向HashSet集合中存入一个元素时,HashSet会调用该对象的hashCode()方法来得到该对象的hashCode值,然后根据该值确定对象在HashSet中的存储位置。在Hash集合中,不能同时存放两个相等的元素,而判断两个元素相等的标准是两个对象通过equals方法比较相等并且两个对象的HashCode方法返回值也相等。

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

5.清空与删除

HashSet提供了remove(Object o)删除元素、clear()清除所有元素的方法。

    //如果指定元素存在于此set中,则将其移除
    public boolean remove(Object o) {
        return map.remove(o)==PRESENT;
    }

    //从此set中移除所有元素
    public void clear() {
        map.clear();
    }

6.其他公开的方法

size()、isEmpty()、clone()

    //返回此set中的元素的数量
    public int size() {
        return map.size();
    }

    //如果此set不包含任何元素,则返回 true
    public boolean isEmpty() {
        return map.isEmpty();
    }

    //返回此 HashSet实例的浅表副本
    @SuppressWarnings("unchecked")
    public Object clone() {
        try {
            HashSet<E> newSet = (HashSet<E>) super.clone();
            newSet.map = (HashMap<E, Object>) map.clone();
            return newSet;
        } catch (CloneNotSupportedException e) {
            throw new InternalError(e);
        }
    }

支持序列化的写入函数writeObject(java.io.ObjectOutputStream s)和读取函数readObject(java.io.ObjectInputStream s):

    //java.io.Serializable的写入函数,将HashSet的“总的容量,加载因子,实际容量,所有的元素”都写入到输出流中
    private void writeObject(java.io.ObjectOutputStream s)
        throws java.io.IOException {
        // Write out any hidden serialization magic
        s.defaultWriteObject();

        // Write out HashMap capacity and load factor
        s.writeInt(map.capacity());
        s.writeFloat(map.loadFactor());

        // Write out size
        s.writeInt(map.size());

        // Write out all elements in the proper order.
        for (E e : map.keySet())
            s.writeObject(e);
    }

    // java.io.Serializable的读取函数,将HashSet的“总的容量,加载因子,实际容量,所有的元素”依次读出
    private void readObject(java.io.ObjectInputStream s)
        throws java.io.IOException, ClassNotFoundException {
        // Read in any hidden serialization magic
        s.defaultReadObject();

        // Read capacity and verify non-negative.
        int capacity = s.readInt();
        if (capacity < 0) {
            throw new InvalidObjectException("Illegal capacity: " +
                                             capacity);
        }

        // Read load factor and verify positive and non NaN.
        float loadFactor = s.readFloat();
        if (loadFactor <= 0 || Float.isNaN(loadFactor)) {
            throw new InvalidObjectException("Illegal load factor: " +
                                             loadFactor);
        }

        // Read size and verify non-negative.
        int size = s.readInt();
        if (size < 0) {
            throw new InvalidObjectException("Illegal size: " +
                                             size);
        }

        // Set the capacity according to the size and load factor ensuring that
        // the HashMap is at least 25% full but clamping to maximum capacity.
        capacity = (int) Math.min(size * Math.min(1 / loadFactor, 4.0f),
                HashMap.MAXIMUM_CAPACITY);

        // Create backing HashMap
        map = (((HashSet<?>)this) instanceof LinkedHashSet ?
               new LinkedHashMap<E,Object>(capacity, loadFactor) :
               new HashMap<E,Object>(capacity, loadFactor));

        // Read in all elements in the proper order.
        for (int i=0; i<size; i++) {
            @SuppressWarnings("unchecked")
                E e = (E) s.readObject();
            map.put(e, PRESENT);
        }
    }

三、HashSet的迭代

 HashSet通过调用HashMap.keySet()返回<key, value>对中的key集以此得到集合的迭代器。

    //返回对此 set中元素进行迭代的迭代器
    public Iterator<E> iterator() {
        return map.keySet().iterator(); //HashMap.keySet()返回<key, value>对中的key集
    }
时间: 2024-08-29 21:58:48

jdk1.8.0_45源码解读——HashSet的实现的相关文章

jdk1.8.0_45源码解读——Set接口和AbstractSet抽象类的实现

jdk1.8.0_45源码解读——Set接口和AbstractSet抽象类的实现 一. Set架构 如上图: (01) Set 是继承于Collection的接口.它是一个不允许有重复元素的集合.(02) AbstractSet 是一个抽象类,它继承于AbstractCollection.AbstractCollection实现了Set中的绝大部分函数,为Set的实现类提供了便利.(03) HastSet 和 TreeSet 是Set的两个实现类.        HashSet依赖于HashMa

jdk1.8.0_45源码解读——HashMap的实现

jdk1.8.0_45源码解读——HashMap的实现 一.HashMap概述 HashMap是基于哈希表的Map接口实现的,此实现提供所有可选的映射操作.存储的是<key,value>对的映射,允许多个null值和一个null键.但此类不保证映射的顺序,特别是它不保证该顺序恒久不变.  除了HashMap是非同步以及允许使用null外,HashMap 类与 Hashtable大致相同. 此实现假定哈希函数将元素适当地分布在各桶之间,可为基本操作(get 和 put)提供稳定的性能.迭代col

jdk1.8.0_45源码解读——Map接口和AbstractMap抽象类的实现

jdk1.8.0_45源码解读——Map接口和AbstractMap抽象类的实现 一. Map架构 如上图:(01) Map 是映射接口,Map中存储的内容是键值对(key-value).(02) AbstractMap 是继承于Map的抽象类,它实现了Map中的大部分API.其它Map的实现类可以通过继承AbstractMap来减少重复编码.(03) SortedMap 是继承于Map的接口.SortedMap中的内容是排序的键值对,排序的方法是通过比较器(Comparator).(04) N

jdk1.8.0_45源码解读——ArrayList的实现

jdk1.8.0_45源码解读——ArrayList的实现 一.ArrayList概述 ArrayList是List接口的可变数组的实现.实现了所有可选列表操作,并允许包括 null 在内的所有元素.除了实现 List 接口外,此类还提供一些方法来操作内部用来存储列表的数组的大小. 每个ArrayList实例都有一个容量,该容量是指用来存储列表元素的数组的大小.它总是至少等于列表的大小.随着向ArrayList中不断添加元素,其容量也自动增长.自动增长会带来数据向新数组的重新拷贝,因此,如果可预

jdk1.8.0_45源码解读——LinkedList的实现

jdk1.8.0_45源码解读——LinkedList的实现 一.LinkedList概述 LinkedList是List和Deque接口的双向链表的实现.实现了所有可选列表操作,并允许包括null值.    LinkedList既然是通过双向链表去实现的,那么它可以被当作堆栈.队列或双端队列进行操作.并且其顺序访问非常高效,而随机访问效率比较低. 注意,此实现不是同步的. 如果多个线程同时访问一个LinkedList实例,而其中至少一个线程从结构上修改了列表,那么它必须保持外部同步.这通常是通

JDK1.8 FutureTask源码解读(Future模式)

在Java中比较常见的两种创建线程的方法:继承Thread类和实现Runnable接口.但是这两种方法有个缺点就是无法获取线程执行后的结果.所以Java之后提供了Future和Runnable接口,用于实现获取线程执行结果.下面开始源码分析: 1.Callable接口 public interface Callable<V> { //返回接口,或者抛出异常 V call() throws Exception; } 2.Future接口 public interface Future<V&

HttpClient 4.3连接池参数配置及源码解读

目前所在公司使用HttpClient 4.3.3版本发送Rest请求,调用接口.最近出现了调用查询接口服务慢的生产问题,在排查整个调用链可能存在的问题时(从客户端发起Http请求->ESB->服务端处理请求,查询数据并返回),发现原本的HttpClient连接池中的一些参数配置可能存在问题,如defaultMaxPerRoute.一些timeout时间的设置等,虽不能确定是由于此连接池导致接口查询慢,但确实存在可优化的地方,故花时间做一些研究.本文主要涉及HttpClient连接池.请求的参数

Spring源码解读之XmlBeanFactory

首先感谢<Spring源码深度解析>郝佳.接下来的Spring源码解读系列,都是读了郝佳的书后的观后感.再次感谢他,带我走进了源码的世界. BeanFactory factory= new XmlBeanFactory (new ClassPathResource("D:\\Project\\Eclipse\\Spring_Maven\\src\\main\\resources\\spring_beans.xml" )); new ClassPathResource(Str

JDK源码解读之Integer(1)

本系列文章使用的JDK版本为jdk1.8.0_131,一些基础的知识储备:原码.反码.补码,移位,建议参考文章:<原码,反码,补码 详解><Java 源码学习系列(三)--Integer> Integer是我们开发过程中最常用的一个类,因此JDK的源码解读就从它开始吧.凡是对Java有点了解的都知道,Integer是int的包装类型,长度为32位.因此我们可以看到如下定义 //可表示的最小值:-2^31,至于为什么是这个数,上面的文章讲的很清楚了 @Native public st