一、HashMap的底层实现

HashMap底层是基于数组和链表实现的。其中最重要的参数:容量和负载因子。

容量的默认大小事16,负载因子是0.75,当HashMap的size>16*0.75的时候就会发生库容(容量和负载因子都可以自由调整)

Hashmap实现了Map接口,允许放入null元素,出了该类未实现同步外,其余和HashTable大致相同,跟TreeMap不同,该容器不保证冤死顺序,根据需要该容器可能对元素重新哈希,元素的顺序也会被重新打散,因此不同时间迭代同一个HashMap的顺序可能会不同。

二、HashMap的put(key,value)方法

首先会将传入的可以、做hash运算计算出hashCode,然后根据数组长度取模计算出在数组中的index下表

由于在计算机中位运算比取模运算效率高,所以HashMsap规定数组的长度为2n。这样用2n-1做位运算与取模效果一致,并且效率要高出许多

由于数组的长度有限,所以难免出现不同放入key通过运算得到的index相同,这种情况可以利用链表来解决,HashMap会在table[index]出形成链表,采用头插法将数据插入链表中

三、HashMap的get(key)fangfa

get和put类似,也是讲传入的可以计算出index,如果该位置上是一个链表就需要比那里整个链表,通过key.equals(k)来找到对应的元素。

遍历方式:

第一种

Iterator<Map.Entry<String, Integer>> entryIterator=map.entrySet().iterator();
while(entryIterator.hasNext()){
Map.Entry<String,Integer> next=entryIterator.next();
System.err.println("key="+next.getKey()+"value="+next.getValue());
}

第二种

Iterator iterator=map.keySet().iterator();
while(iterator.hasNext()){
String key=iterator.next();
System.err.println("key="+key+"value="+map.get(key));
}

第三种

map.forEach((key,value)->{
System.err.println("key="+key+"value="+value);
});

第一种可以把key value同时取出,第二种还得需要通过key去一次value,效率较低,第三种需要JDK1.8以上,通过外层遍历table,内层遍历链表或红黑树。

四、为什么多线程场景下不推荐使用HashMap

在并发环境下使用HashMap容易出现死循环。并发场景下发生扩容,调用resize()方法里的rehash()时,容易出现环形链表。这样当获取一个不存在的key时,计算出的index正好是环形链表的下标时就会出现死循环

所以,HashMap只能在单线程中使用,并且尽量的预设容量,尽可能的减少扩容发

在JDK1.8中对HashMap进行了优化:当hash碰撞之后写入链表的长度超过阈值(默认为8),链表将会转换成红黑树。假设hash冲突非常严重,一个数组后面接了很长的链表,此时查询的时间复杂度就是O(n)。如果是红黑树,时间复杂度就是O(logn)。大大提高了查询的效率。多线程场景下推荐使用ConcurrentHashMap。

五、HashSet的底层实现

HashSet是对HashMap的简单包装,对HashSet的函数调用都会转换成合适的HashMap方法,因此HashSet的实现非常简单。

成员变量

首先了解下HashSet的成员变量

  private transient HashMap<E,Object> map;

    // Dummy value to associate with an Object in the backing Map
private static final Object PRESENT = new Object();

发现主要有两个变量:

map:用于存放最终数据

PRESENT:是所有写入map的value值

构造函数

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

构造函数很简单,利用了HashMap初始化了map

add

 public boolean add(E e) {
return map.put(e, PRESENT)==null;
}

比较关键的就是这个add()方法。可以看出他是将存放的对象当做了HashMap的键,value都是相同的PRESENT.由于HashMap的key是不能重复的,所以每当有重复的值写入到HashSet中只能存放不重复的元素

最新文章

  1. CEUtils---我在Unity中使用的一些小类库(不断更新中)
  2. 直接请求URL调用 axis webservices
  3. 读书笔记--编程珠玑II
  4. Numpy矩阵取列向量
  5. 安装 nodejs图像处理模块 sharp
  6. CentOS7安装telnet服务
  7. smarty安装及例子
  8. C语言关键字-volatile
  9. NVL函数(NVL,NVL2,NULLIF,COALESCE)
  10. find详解
  11. OC语法1——OC概述
  12. encode_utf8 把字符编码成字节 decode_utf8解码UTF-8到字符
  13. Java读取本地文件,并显示在JSP文件中
  14. MonkeyRecorder
  15. .NET Core中的包、元包与框架
  16. 微信小程序页面无法跳转
  17. /bin/ls: Permission denied
  18. P2245 星际导航
  19. Python开发【数据结构】:基础
  20. BZOJ4689 Find the Outlier 【高斯消元】*

热门文章

  1. 转载------一小时包教会 —— webpack 入门指南
  2. 「疫期集训day1」无言
  3. 树的深度———树形DP
  4. 不用破解版的 Navicat 了,几款免费且好用的 SQL 客户端送给你
  5. 浏览器如何解析css选择器?
  6. 最大熵原理(The Maximum Entropy Principle)
  7. Static关键字的使用
  8. Andriod 自动化环境搭建
  9. SpringBoot整合Swagger3生成接口文档
  10. DJANGO-天天生鲜项目从0到1-002-用户模块-注册