HashSet源码分析以及自定义HashSet学习
2018-03-20 15:50
513 查看
1.源码学习:
public class HashSet<E> extends AbstractSet<E> implements Set<E>, Cloneable, java.io.Serializable { 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); } /** * 底层实际调用底层HashMap的keySet来返回所有的key。 */ public Iterator<E> iterator() { return map.keySet().iterator(); } /** * 底层实际调用HashMap的size()方法返回Entry的数量,就得到该Set中元素的个数。 */ public int size() { return map.size(); } /** * 底层实际调用HashMap的isEmpty()判断该HashSet是否为空。 public boolean isEmpty() { return map.isEmpty(); } /** * 底层实际调用HashMap的containsKey判断是否包含指定key。 */ 4000 public boolean contains(Object o) { return map.containsKey(o); } /** * * 新元素作为key放入, 通过map的key不可重复,来实现set的不可重复特点 */ public boolean add(E e) { return map.put(e, PRESENT)==null; } /** * 底层实际调用HashMap的remove方法删除指定Entry。 */ public boolean remove(Object o) { return map.remove(o)==PRESENT; } /** * 底层实际调用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(); } } }
2.自定义HashSet:
public class SxtHashSet { HashMap map; private static final Object PRESENT = new Object(); public SxtHashSet(){ map = new HashMap(); } public int size(){ return map.size(); } public void add(Object o){ map.put(o, PRESENT); //set的不可重复就是利用了map里面键对象的不可重复! } public static void main(String[] args) { SxtHashSet s = new SxtHashSet(); s.add("aaa"); s.add(new String("aaa")); System.out.println(s.size()); } }
相关文章推荐
- Combres库 学习小结以及部分源码分析
- Okhttp缓存源码分析以及自定义缓存实现
- Okhttp缓存源码分析以及自定义缓存实现
- spring mvc框架源码分析(二)-自定义注解以及通过反射获取注解
- 操盘机器人新增板块资金流向分析模型,可自定义板块、分析周期以及系统自动学习功能
- 为什么HashSet不能重复以及具体原理源码分析
- Spark组件之GraphX学习5--随机图生成和消息发送aggregateMessages以及mapreduce操作(含源码分析)
- 操盘机器人新增板块资金流向分析模型,可自定义板块、分析周期以及系统自动学习功能
- Android开发学习之路-RecyclerView的Item自定义动画及DefaultItemAnimator源码分析
- spark源码学习(三):job的提交以及runJob函数的分析
- Combres库 学习小结以及部分源码分析
- java学习之ArrayList和HashSet的比较以及HashCode分析
- java核心基础--jdk源码分析学习--HashSet
- hadoop学习之路(二)hadoop基本概念原理以及单词统计任务源码分析
- cocos2D-X源码分析之从cocos2D-X学习OpenGL(3)----BATCH_COMMAND
- android smack源码分析——接收消息以及如何解析消息
- nginx源码_分析学习(一)_开始情绪篇
- Android学习——maven插件的安装以及关联源码的几种方式总结
- struts2源码分析--IOC容器的实现(操作以及容器的初始化)
- 仿美团项目学习源码分析(3)--数组加key