HashMap的源码分析
hashMap的底层实现是 数组+链表 的数据结构,数组是一个Entry<K,V>[] 的键值对对象数组,在数组的每个索引上存储的是包含Entry的节点对象,每个Entry对象是一个单链表结构,维护这下一个Entry节点的引用;有点绕,用个图来展示吧:

Entry<K,V>[] 数组部分保存的是首个Entry节点;Entry节点包含一个 K值引用 V值引用 以及 引用下一个Entry 节点的next引用;
Entry节点的java代码实现如下:
static class Entry<K,V> implements Map.Entry<K,V> {
        final K key;    //key 引用
        V value;         //value  引用
        Entry<K,V> next;   //下一个Entry 节点的引用
}
下面再看下HashMap 对象的java实现代码:
包含的属性有:
public class HashMap<K,V>
extends AbstractMap<K,V>
implements Map<K,V>, Cloneable, Serializable
{ /**
* The default initial capacity - MUST be a power of two.
*/
static final int DEFAULT_INITIAL_CAPACITY = 1 << 4; // aka 16 /**
* The maximum capacity, used if a higher value is implicitly specified
* by either of the constructors with arguments.
* MUST be a power of two <= 1<<30.
*/
static final int MAXIMUM_CAPACITY = 1 << 30; /**
* The load factor used when none specified in constructor.
*/
static final float DEFAULT_LOAD_FACTOR = 0.75f; /**
* An empty table instance to share when the table is not inflated.
*/
static final Entry<?,?>[] EMPTY_TABLE = {}; /**
* The table, resized as necessary. Length MUST Always be a power of two.
*/
transient Entry<K,V>[] table = (Entry<K,V>[]) EMPTY_TABLE; /**
* The number of key-value mappings contained in this map.
*/
transient int size; /**
* The next size value at which to resize (capacity * load factor).
* @serial
*/
// If table == EMPTY_TABLE then this is the initial capacity at which the
// table will be created when inflated.
int threshold; /**
* The load factor for the hash table.
*
* @serial
*/
final float loadFactor;
}
比较重要的属性是:
transient Entry<K,V>[] table = (Entry<K,V>[]) EMPTY_TABLE; 表明这是一个 Entry<K,V>[] 的数组类型;
下面看其无参的构造器:
public HashMap() {
        this(DEFAULT_INITIAL_CAPACITY, DEFAULT_LOAD_FACTOR);
    }
进入以下的构造方法:
 public HashMap(int initialCapacity, float loadFactor) {        //initialCapacity:16   loadFactor 0.75f
        if (initialCapacity < 0)
            throw new IllegalArgumentException("Illegal initial capacity: " +
                                               initialCapacity);
        if (initialCapacity > MAXIMUM_CAPACITY)     //MAXIMUM_CAPACITY   1073741824  false
initialCapacity = MAXIMUM_CAPACITY;
if (loadFactor <= 0 || Float.isNaN(loadFactor)) false
throw new IllegalArgumentException("Illegal load factor: " +
loadFactor); this.loadFactor = loadFactor; //赋值给loadFactor=0.75
threshold = initialCapacity; //赋值给threshold=16 当为16是自动扩容
init();
}
下面再看看put(E e)的方法:
public V put(K key, V value) {    //如插入  key="city"   value="shanghai"
        if (table == EMPTY_TABLE) {     //true
            inflateTable(threshold);   //参数为16
        }
        if (key == null)        // false
            return putForNullKey(value);
        int hash = hash(key);    //返回key值的hash码; 比如返回为 337
        int i = indexFor(hash, table.length);  //将hash 取模与16 获得的结果为 1
        for (Entry<K,V> e = table[i]; e != null; e = e.next) {  //遍历 Entry[1] 中的链表节点对象 包含 原先有的节点和新增进去的节点
            Object k;
            if (e.hash == hash && ((k = e.key) == key || key.equals(k))) {  //当Entry中包含 相同的hash码的key 并且key和要添加的key相等即可以是否重复  则进入以下逻辑:新节点替换重复的节点
                V oldValue = e.value;
                e.value = value;
                e.recordAccess(this);
                return oldValue;
            }
        }
下面是 inflateTable(threshold) 方法的源码;
/**
* Inflates the table.
*/
private void inflateTable(int toSize) { //toSize 16
// Find a power of 2 >= toSize
int capacity = roundUpToPowerOf2(toSize); //capacity=16 threshold = (int) Math.min(capacity * loadFactor, MAXIMUM_CAPACITY + 1); // threshold=16*0.75
table = new Entry[capacity]; //创建 Entry[] 数组长度为16
initHashSeedAsNeeded(capacity); //这个方法可以暂时不用深究
}
下面是 roundUpToPowerOf2(int i) 源码
private static int roundUpToPowerOf2(int number) {    //  number=16
        // assert number >= 0 : "number must be non-negative";
        return number >= MAXIMUM_CAPACITY   //fase  返回 16
                ? MAXIMUM_CAPACITY
                : (number > 1) ? Integer.highestOneBit((number - 1) << 1) : 1;   //number=16>1  返回 16
    } 
put 方法的源码分析完了之后,接下来再看一下get(Object key) 的方法; 源码:
public V get(Object key) {    //如 key="name"
        if (key == null)      //false
            return getForNullKey();
        Entry<K,V> entry = getEntry(key);
        return null == entry ? null : entry.getValue();
    }
 final Entry<K,V> getEntry(Object key) {   //key=name
        if (size == 0) {    //false
            return null;
        }
        int hash = (key == null) ? 0 : hash(key);   //例如 返回hash=337
        for (Entry<K,V> e = table[indexFor(hash, table.length)];  //indexFor(hash, table.length)上面分析过 返回值为 1;遍历 table[1] 中的节点
             e != null;
             e = e.next) {
            Object k;
            if (e.hash == hash &&
                ((k = e.key) == key || (key != null && key.equals(k))))   //如果存在key的hash码相等,并且对象也相等则返回 对应的Entry 节点
                return e;
        }
        return null;   //否则返回null
    } 
到此,hashMap 的源码基本分析完毕了,通过源码分析我们知道HashMap的底层是 数组+链表结构来存数数据的,添加节点存储的位置是根据 key 取hash值 再取模于数组长度:返回的数值就是Entry接在在数组的哪个位置;这种方式的存储方式减少了存储的时间和空间的复杂度;
知道了hashMap是由 数组+链表 的数据结构存储数据后,我们也很容易明白hashMap 的遍历方式:
HashMap的源码分析的更多相关文章
- HashMap的源码分析与实现  伸缩性角度看hashmap的不足
		本文介绍 1.hashmap的概念 2.hashmap的源码分析 3.hashmap的手写实现 4.伸缩性角度看hashmap的不足 一.HashMap的概念 HashMap可以将其拆分为Hash散列 ... 
- Java——HashMap底层源码分析
		1.简介 HashMap 根据键的 hashCode 值存储数据,大多数情况下可以直接定位到它的值,因而具有很快的访问速度,但遍历顺序却是不确定的. HashMap 最多只允许一条记录的key为 nu ... 
- Java中HashMap的源码分析
		先来回顾一下Map类中常用实现类的区别: HashMap:底层实现是哈希表+链表,在JDK8中,当链表长度大于8时转换为红黑树,线程不安全,效率高,允许key或value为null HashTable ... 
- HashMap方法源码分析
		本文将分析put(),resize(),get()和remove()方法的源码 putval()方法 大致步骤:计算key的hash值:根据hash值计算数组下标:判断下标处是否有节点,无节点则直接插 ... 
- Java源码——HashMap的源码分析及原理学习记录
		学习HashMap时,需要带着这几个问题去,会有很大的收获: 一.什么是哈希表 二.HashMap实现原理 三.为何HashMap的数组长度一定是2的次幂? 四.重写equals方法需同时重写hash ... 
- HashMap LinkedHashMap源码分析笔记
		MapClassDiagram 
- Java HashMap实例源码分析
		引言 HashMap在键值对存储中被经常使用,那么它到底是如何实现键值存储的呢? 一 Entry Entry是Map接口中的一个内部接口,它是实现键值对存储关键.在HashMap中,有Entry的实现 ... 
- HashMap的源码分析(一)
		1.hashMap的关键值 DEFAULT_INITIAL_CAPACITY:默认初始容量16,∈(0,1<<30),实际大小为2的整数次幂: DEFAULT_LOAD_FACTOR:默认 ... 
- HashMap从源码分析数据结构
		1. HashMap在链表中存储的是键值对 2. 数组是一块连续的固定长度的内存空间,再好的哈希函数也不能保证得到的存储地址绝对不发生冲突.那么哈希冲突如何解决呢?哈希冲突的解决方案有多种:开放定址法 ... 
随机推荐
- linux学习资料收藏
			http://blog.chinaunix.net/uid/10167808/abstract/1.html?year=2008 http://linux.linuxidc.com/ind ... 
- ceph结构详解
			引言 那么问题来了,把一份数据存到一群Server中分几步? Ceph的答案是:两步. 计算PG 计算OSD 计算PG 首先,要明确Ceph的一个规定:在Ceph中,一切皆对象. 不论是视频,文本,照 ... 
- 记账本,C,Github,util
			package util; import java.awt.Component; import java.awt.Dimension; import javax.swing.JButton; impo ... 
- [leetcode]55. Jump Game青蛙跳(能否跳到终点)
			Given an array of non-negative integers, you are initially positioned at the first index of the arra ... 
- 利用Sharding-Jdbc实现分表[z]
			[z]https://www.cnblogs.com/codestory/p/5591651.html 你们团队使用SpringMVC+Spring+JPA框架,快速开发了一个NB的系统,上线后客户订 ... 
- message [Failed to convert property value of type [java.lang.String] to required type [java.util.Date] for property
			springmvc前台字符串,后台Date类型字段.时间强转失败 数值:18年12月31日 15:43:21 解决方法,给时间字段加注释 @DateTimeFormat(pattern = " ... 
- 用TSQL从sqlserve 发布订阅链中删除一张或几张表
			一个简单的存储过程,用来实现从一个SQLSERVE 发布订阅链中删除一张或几张表. /* 1.停日志读取代理 2.exec usp_从复制订阅中删除表 'dbtestPub','test1' 3.开日 ... 
- P1495 曹冲养猪(拓展欧几里得)
			题目描述 自从曹冲搞定了大象以后,曹操就开始捉摸让儿子干些事业,于是派他到中原养猪场养猪,可是曹冲满不高兴,于是在工作中马马虎虎,有一次曹操想知道母猪的数量,于是曹冲想狠狠耍曹操一把.举个例子,假如有 ... 
- GUI学习之一——PyQt5初识
			我们在第〇篇里先演示了GUI的功能,其实Python有多个库是支持GUI编程的,python官网列出了大量的说明,其中包括了原生的tkinter 还有许多第三方库 Pyqt PySide wxPyth ... 
- java 多线程通知 CountDownLatch 倒数计数器的使用
			package com.hra.riskprice; import com.hra.riskprice.SysEnum.Factor_Type; import org.springframework. ... 
