Style:Mac

Series:Java

Since:2018-09-10

End:2018-09-10

Total Hours:1

Degree Of Diffculty:5

Degree Of Mastery:5

Practical Level:5

Desired Goal:5

Archieve Goal:3

Gerneral Evaluation:3

Writer:kingdelee

Related Links:

http://www.cnblogs.com/kingdelee/

http://www.runoob.com/java/java-operators.html

1.解析常见的HaspMap操作背后的故事

package jdk11.map;

import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger; public class Client { private static final Logger logger = LogManager.getLogger(Client.class); public static void main(String[] args) { Map<Father, Integer> map = new HashMap<>();
Father f1 = new Father("f1");
f1.setHashcode(1);
Father f2 = new Father("f2");
f2.setHashcode(1);
Father f3 = new Father("f3");
f2.setHashcode(1);
Father f4 = new Father("f4");
f2.setHashcode(1);
Father f5 = new Father("f5");
f2.setHashcode(1);
Father f6 = new Father("f6");
f2.setHashcode(1);
Father f7 = new Father("f7");
f2.setHashcode(1);
Father f8 = new Father("f8");
f2.setHashcode(1);
Father f9 = new Father("f9");
f2.setHashcode(1); map.put(f1, 1);
map.put(f2, 2);
map.put(f3, 3);
map.put(f4, 4);
map.put(f5, 5);
map.put(f6, 6);
map.put(f7, 7);
map.put(f8, 8);
map.put(f9, 9);
map.put(new Father("10", 1), 10);
map.put(new Father("11", 1), 11);
map.put(new Father("12", 1), 12);
map.put(new Father("13", 1), 13);
map.put(new Father("14", 1), 14);
map.put(new Father("15", 1), 15);
map.put(new Father("16", 1), 16);
map.put(new Father("17", 1), 17);
map.put(new Father("18", 1), 18);
map.put(new Father("19", 1), 19);
map.put(new Father("20", 1), 20);
map.put(new Father("21", 1), 21);
logger.info(map.get(f1));
logger.info(map.get(f2));
logger.info(map.get(new Father("21", 1))); }
} class Father {
public String name; public int hashcode; public Father(String name, int hashcode) {
this.name = name;
this.hashcode = hashcode;
} public Father(String name) {
this.name = name;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public int getHashcode() {
return hashcode;
} public void setHashcode(int hashcode) {
this.hashcode = hashcode;
} @Override
public int hashCode() {
return getHashcode();
} @Override
public boolean equals(Object obj) {
return name.equals(((Father) obj).name) && hashcode == (((Father) obj).hashcode);
}
}

  

2.new

很简单,仅仅对一些常量做初始化其中比较重要的有

public HashMap() {
this.loadFactor = DEFAULT_LOAD_FACTOR;
}

  

loadFactor=0.75,扩展因子

DEFAULT_INITIAL_CAPACITY=16,初始化的容量为16

....

3.put

3.1 算出hash()

 public V put(K key, V value) {
logger.info("put---------------------modCount:" + modCount);
return putVal(hash(key), key, value, false, true);
}

  

3.2 putVal()

声明Node[] tab,以便该数组类型做横向数组填充

进行多个判断:

当tab为空时,会创建tab,并给tab重新构建空间resize()

当需要填坑的坑位是空时,横向创建坑位

如果需要填的坑已被占,则考虑纵向创建该坑位关联节点;

考虑情况有3,1.如果是指向同一个内存地址则忽略操作;2.如果需要创建的链式节点已经进化成红黑树,则执行插入红黑树的操作;3.还不是红黑树,则执行该节点的链表创建关联链式节点操作

以上是宏观上看待put。

3.2.1 微观上看putVal()

3.2.1.1 即tab未创建时,通过resize()创建tab

if ((tab = table) == null || (n = tab.length) == 0) {
logger.info("table为null");
n = (tab = resize()).length; // 1.当未指定初始容量时,进行resize, 得到容量值赋给n=16; 获得新的节点给tab;已经存在节点时不再进来
logger.info("tab renTab");
}

如果要put的坑位是空的,就将创建该坑位Node  

if ((p = tab[i = (n - 1) & hash]) == null)  // i: (16-1) & 10 = 10,未存在节点的情况下,让新节点P指向数组节点tab中的hash后的节点,创建节点数组;已经存在节点时不再进来
{
logger.info("创建一个新节点,tab["+i+"]指向这个节点" + "hash:" + hash + ",value:" + value);
tab[i] = newNode(hash, key, value, null); // 仅在p节点为空的情况下,创建刚刚新节点指向hash后为空的节点的位置
}

3.2.1.2 否则执行一段非常复杂的逻辑

3.2.1.2.1   

判断是否为指向统一内存地址,如果是,则break

 Node<K,V> e; K k;
if (p.hash == hash && ((k = p.key) == key || (key != null && key.equals(k)))){
e = p; // hash相等 && key相等 的情况下,用节点e存储原来的已经存在的节点k
logger.info("相同对象");
}

  

不是同一内存地址,继续判断,需要该坑位是否已经进化成红黑树,如果是,就执行红黑树的put,不是则继续判断

else if (p instanceof TreeNode)     // 是否已经进化成红黑树
{
logger.info("进化红黑树");
e = ((TreeNode<K,V>)p).putTreeVal(this, tab, hash, key, value);
}

以上条件都不是,说明这货要以链表或者首次进化成树的方式put进来

通过for依次遍历 坑位的链表(  以这个坑位为首索引,通过next连接的各个子节点  )

如果该坑位或者坑位的子节点为空,则为其创建新的节点,形成链表形式;紧接着判断子节点数(算上坑位)是否>=7,即链表串元素>=7时,进行put会进行treeifyBin()进化成红黑树,不满足则break

如果需要put的这货在坑位的子节点中已经存在了,break

  else {
logger.info("发生碰撞, 当前p.hash:" + p.hash + ",p.value:" + value + ",进入搜索next节点的循环");
for (int binCount = 0; ; ++binCount) {
if ((e = p.next) == null) {
logger.info("p.next为空,为其创建新的节点,p.next.hash:" + hash + ", p.next.value:" + value);
p.next = newNode(hash, key, value, null); // 将当前的节点的下一个节点指向新创建的节点
if (binCount >= TREEIFY_THRESHOLD - 1) // 只有>=7次迭代才会执行进化树结构
{
logger.info("binCount >= (TREEIFY_THRESHOLD - 1), binCount:" + binCount + ", (TREEIFY_THRESHOLD - 1):" + (TREEIFY_THRESHOLD - 1));
treeifyBin(tab, hash);
}
logger.info("跳出循环");
break;
}
if (e.hash == hash && ((k = e.key) == key || (key != null && key.equals(k)))){
logger.info("同一个对象");
break;
}
logger.info("p.next有值, p.next.hash:" + p.next.hash + ",p.next.value:" + p.next.value + ", 把当前p指针指向p.next");
p = e;
}
}

  

最后,put成功的话会count++,判断是否需要resize,结束put。

++modCount; //执行put操作的次数
logger.info("modCount:" + modCount);
if (++size > threshold) //已经存放元素的容量+1 与 扩容阀值进行对比
{
logger.info("++size > threshold, size:" + size + ", threshold:" + threshold);
resize();
}

  

3.2.1.3 判断是否需要扩容

3.2.1.1 When&How执行 resize()

3.2.1.2 When&How进行链式或者红黑树操作

3.3 get()

【JVM】-NO.113.JVM.1 -【JDK11 HashMap详解-0-全局-put】的更多相关文章

  1. 【JVM】-NO.113.JVM.1 -【JDK11 HashMap详解-4-resize()】

    Style:Mac Series:Java Since:2018-09-10 End:2018-09-10 Total Hours:1 Degree Of Diffculty:5 Degree Of ...

  2. 【JVM】-NO.110.JVM.1 -【JDK11 HashMap详解】

    Style:Mac Series:Java Since:2018-09-10 End:2018-09-10 Total Hours:1 Degree Of Diffculty:5 Degree Of ...

  3. 【JVM】-NO.114.JVM.1 -【JDK11 HashMap详解-3-put-treeifyBin()-AVL】

    Style:Mac Series:Java Since:2018-09-10 End:2018-09-10 Total Hours:1 Degree Of Diffculty:5 Degree Of ...

  4. 【JVM】-NO.115.JVM.1 -【JDK11 HashMap详解-4-伸展树、B树】

    .Style:Mac Series:Java Since:2018-09-10 End:2018-09-10 Total Hours:1 Degree Of Diffculty:5 Degree Of ...

  5. 【JVM】-NO.116.JVM.1 -【JDK11 HashMap详解-5-红黑树】

    Style:Mac Series:Java Since:2018-09-10 End:2018-09-10 Total Hours:1 Degree Of Diffculty:5 Degree Of ...

  6. 【JVM】-NO.111.JVM.1 -【JDK11 HashMap详解-1-hash()剖析】

    Style:Mac Series:Java Since:2018-09-10 End:2018-09-10 Total Hours:1 Degree Of Diffculty:5 Degree Of ...

  7. 【JVM】-NO.112.JVM.2 -【JDK11 HashMap详解-2-tab[i = (n - 1) & hash])剖析】

    Style:Mac Series:Java Since:2018-09-10 End:2018-09-10 Total Hours:1 Degree Of Diffculty:5 Degree Of ...

  8. java面试题之----JVM架构和GC垃圾回收机制详解

    JVM架构和GC垃圾回收机制详解 jvm,jre,jdk三者之间的关系 JRE (Java Run Environment):JRE包含了java底层的类库,该类库是由c/c++编写实现的 JDK ( ...

  9. 【转】 java中HashMap详解

    原文网址:http://blog.csdn.net/caihaijiang/article/details/6280251 java中HashMap详解 HashMap 和 HashSet 是 Jav ...

随机推荐

  1. 使用ffmpeg搭建HLS直播系统

    [时间:2018-04] [状态:Open] [关键词:流媒体,stream,HLS, ffmpeg,live,直播,点播, nginx, ssegment] 0 引言 本文作为HLS综述的后续文章. ...

  2. spring boot 项目启动无任何反应

    遇到的问题 spring boot项目启动后无任何报错,ps有进程,nohub无日志 定位 更换jar包,问题依然存在,将jar包放到其他服务器,运行正常,排除打包问题 同服务器其他系统运行正常,但停 ...

  3. 嵌入式开发之zynqMp ---Zynq UltraScale+ MPSoC 图像编码板zcu102

    1.1 xilinx zynqMp 架构 1.1.1 16nm 级别工艺 Zynq UltraScale+  MPSoC架构 Xilinx新一代Zynq针对控制.图像和网络应用推出了差异化的产品系,这 ...

  4. Java Spring MVC 错误 及 常见问题 总结

    [参考]spring入门常见的问题及解决办法 错误: 从Git新获取项目 运行出现 1.org.springframework.beans.factory.BeanDefinitionStoreExc ...

  5. org.hibernate.InvalidMappingException: Could not parse mapping document from无法创建sessionFactory

    把 "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd" 改为 "http://hibernate.sourc ...

  6. 内建模块 datetime使用

    #-*- coding:utf-8 -*- from datetime import datetime #获取当前日期和时间 now =datetime.now() print now # 2017- ...

  7. python机器学习包 Windows下 pip安装 scikit-learn numpy scipy

    1.到PIP的目录中C:\Python34\Scripts;2. 2.1  pip安装numpy pip install numpy 2.2  pip安装sklearn pip install -U ...

  8. python中基本类型的连接组合和互相转换

    本篇总结了一下字符串,列表,字典,元组的连接组合使用和类型的互相转换小例子,尤其列表中的extend()方法和字典中的 update方法非常的常用. 1.连接两个字符串 a = "hello ...

  9. 【thinkphp5】安全建议:隐藏后台登录入口地址

    我们都知道后台   www.test.com/admin 是我们最常用的登录入口,方便的同时也留下了隐患,如果你刚好使用了 admin/ 这种账号密码的方式,会导致我们的后台完全暴露在外. 因此我们建 ...

  10. C# windows GDI+仿画图 绘图程序设计

    C# windows GDI+仿画图 绘图程序设计 1.介绍 这里分享一个简单的画图程序 原作者:author: ping3108@163.com 2.程序主窗体设计 3.程序设计 本程序工程使用VS ...