HashSet底层原理

什么是HashSet

HashSet实现了Set接口,它不允许集合中有重复的值,当我们提到HashSet时,第一件事情就是在将对象存储在HashSet之前,要先确保对象重写equals()和hashCode()方法,这样才能比较对象的值是否相等,以确保set中没有储存相等的对象。如果我们没有重写这两个方法,将会使用这个方法的默认实现。

HashSet类,是存在于java.util包中的类 [1] 。同时也被称为集合,该容器中只能存储不重复的对象

已知实现接口有:Serializable, Cloneable, Iterable, Collection, Set
直接已知子类:JobStateReasons, LinkedHashSet

构造方法

HashSet()
构造一个新的空 set,其底层 HashMap 实例的默认初始容量是 16,加载因子是 0.75。

public HashSet() {
        map = new HashMap<>();
    }

 构造一个包含指定 collection 中的元素的新 set。

public HashSet(Collection<? extends E> c) {
        map = new HashMap<>(Math.max((int) (c.size()/.75f) + 1, 16));
        addAll(c);
    }

构造一个新的空 set,其底层 HashMap 实例具有指定的初始容量和默认的加载因子(0.75)

public HashSet(int initialCapacity) {
        map = new HashMap<>(initialCapacity);
    }

构造一个新的空 set,其底层 HashMap 实例具有指定的初始容量和指定的加载因子

public HashSet(int initialCapacity, float loadFactor) {
        map = new HashMap<>(initialCapacity, loadFactor);
    }

方法摘要

boolean add(E e)
如果此 set 中尚未包含指定元素,则添加指定元素。
void clear()
从此 set 中移除所有元素。
Object clone()
返回此 HashSet 实例的浅表副本:并没有复制这些元素本身。
boolean contains(Object o)
如果此 set 包含指定元素,则返回 true。
boolean isEmpty()
如果此 set 不包含任何元素,则返回 true。
Iterator iterator()
返回对此 set 中元素进行迭代的迭代器。
boolean remove(Object o)
如果指定元素存在于此 set 中,则将其移除。
int size()
返回此 set 中的元素的数量(set 的容量)。
从类 java.util.AbstractSet 继承的方法
equals, hashCode, removeAll
从类 java.util.AbstractCollection 继承的方法
addAll, containsAll, retainAll, toArray, toArray, toString
从类 java.lang.Object 继承的方法
finalize, getClass, notify, notifyAll, wait, wait, wait
从接口 java.util.Set 继承的方法
addAll, containsAll, equals, hashCode, removeAll, retainAll, toArray, toArray
 

实现举例

public class HashSet<E>  
   extends AbstractSet<E>  
   implements Set<E>, Cloneable, java.io.Serializable  
4{  
   static final long serialVersionUID = -5024744406713321676L;  

   // 底层使用HashMap来保存HashSet中所有元素。  
   private transient HashMap<E,Object> map;  

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

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

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

   /** 
    * 以指定的initialCapacity和loadFactor构造一个空的HashSet。 
    * 
    * 实际底层以相应的参数构造一个空的HashMap。 
    * @param initialCapacity 初始容量。 
    * @param loadFactor 加载因子。 
    */  
   public HashSet(int initialCapacity, float loadFactor) {  
   map = new HashMap<E,Object>(initialCapacity, loadFactor);  
   }  

   /** 
    * 以指定的initialCapacity构造一个空的HashSet。 
    * 
    * 实际底层以相应的参数及加载因子loadFactor为0.75构造一个空的HashMap。 
    * @param initialCapacity 初始容量。 
    */  
   public HashSet(int initialCapacity) {  
   map = new HashMap<E,Object>(initialCapacity);  
   }  

   /** 
    * 以指定的initialCapacity和loadFactor构造一个新的空链接哈希集合。 
    * 此构造函数为包访问权限,不对外公开,实际只是是对LinkedHashSet的支持。 
    * 
    * 实际底层会以指定的参数构造一个空LinkedHashMap实例来实现。 
    * @param initialCapacity 初始容量。 
    * @param loadFactor 加载因子。 
    * @param dummy 标记。 
    */  
   HashSet(int initialCapacity, float loadFactor, boolean dummy) {  
   map = new LinkedHashMap<E,Object>(initialCapacity, loadFactor);  
   }  

   /** 
    * 返回对此set中元素进行迭代的迭代器。返回元素的顺序并不是特定的。 
    *  
    * 底层实际调用底层HashMap的keySet来返回所有的key。 
    * 可见HashSet中的元素,只是存放在了底层HashMap的key上, 
    * value使用一个static final的Object对象标识。 
    * @return 对此set中元素进行迭代的Iterator。 
    */  
   public Iterator<E> iterator() {  
   return map.keySet().iterator();  
   }  

   /** 
    * 返回此set中的元素的数量(set的容量)。 
    * 
    * 底层实际调用HashMap的size()方法返回Entry的数量,就得到该Set中元素的个数。 
    * @return 此set中的元素的数量(set的容量)。 
    */  
   public int size() {  
   return map.size();  
   }  

   /** 
    * 如果此set不包含任何元素,则返回true。 
    * 
    * 底层实际调用HashMap的isEmpty()判断该HashSet是否为空。 
    * @return 如果此set不包含任何元素,则返回true。 
    */  
   public boolean isEmpty() {  
   return map.isEmpty();  
   }  

   /** 
    * 如果此set包含指定元素,则返回true。 
    * 更确切地讲,当且仅当此set包含一个满足(o==null ? e==null : o.equals(e)) 
    * 的e元素时,返回true。 
    * 
    * 底层实际调用HashMap的containsKey判断是否包含指定key。 
    * @param o 在此set中的存在已得到测试的元素。 
    * @return 如果此set包含指定元素,则返回true。 
    */  
   public boolean contains(Object o) {  
   return map.containsKey(o);  
   }  

   /** 
    * 如果此set中尚未包含指定元素,则添加指定元素。 
    * 更确切地讲,如果此 set 没有包含满足(e==null ? e2==null : e.equals(e2)) 
    * 的元素e2,则向此set 添加指定的元素e。 
    * 如果此set已包含该元素,则该调用不更改set并返回false。 
    * 
    * 底层实际将将该元素作为key放入HashMap。 
    * 由于HashMap的put()方法添加key-value对时,当新放入HashMap的Entry中key 
    * 与集合中原有Entry的key相同(hashCode()返回值相等,通过equals比较也返回true), 
    * 新添加的Entry的value会将覆盖原来Entry的value,但key不会有任何改变, 
    * 因此如果向HashSet中添加一个已经存在的元素时,新添加的集合元素将不会被放入HashMap中, 
    * 原来的元素也不会有任何改变,这也就满足了Set中元素不重复的特性。 
    * @param e 将添加到此set中的元素。 
    * @return 如果此set尚未包含指定元素,则返回true。 
    */  
   public boolean add(E e) {  
   return map.put(e, PRESENT)==null;  
   }  

   /** 
    * 如果指定元素存在于此set中,则将其移除。 
    * 更确切地讲,如果此set包含一个满足(o==null ? e==null : o.equals(e))的元素e, 
    * 则将其移除。如果此set已包含该元素,则返回true 
    * (或者:如果此set因调用而发生更改,则返回true)。(一旦调用返回,则此set不再包含该元素)。 
    * 
    * 底层实际调用HashMap的remove方法删除指定Entry。 
    * @param o 如果存在于此set中则需要将其移除的对象。 
    * @return 如果set包含指定元素,则返回true。 
    */  
   public boolean remove(Object o) {  
   return map.remove(o)==PRESENT;  
   }  

   /** 
    * 从此set中移除所有元素。此调用返回后,该set将为空。 
    * 
    * 底层实际调用HashMap的clear方法清空Entry中所有元素。 
    */  
   public void clear() {  
   map.clear();  
   }  

   /** 
    * 返回此HashSet实例的浅表副本:并没有复制这些元素本身。 
    * 
    * 底层实际调用HashMap的clone()方法,获取HashMap的浅表副本,并设置到  HashSet中。 
    */  
   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();  
       }  
   }  
}

 

 

 

 

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值