HashMap工作原理

相信大家不管是在Java还是安卓面试过程中,或多或少都会被问及HashMap的工作原理,小编今天大概看了一下Android中HashMap的源码,将结果整理如下,如有不对之处请批评指正:

一、HashMap的数据结构

其实HashMap的存储数据结构是一个散列数组+链表的数据结构,如图:

HashMap的存储数据结构

我们都知道往HashMap里存储值时会传入key和value,HashMap首先会拿到key相对应的hash值,
接着通过hash值计算存放数组的下标,再将key-value对象存放在数组对应下标下的链表里。

接下来我们就根据源码看看HashMap的存取实现。

二、HashMap的存取实现

1) put(key,value)
@Override 
public V put(K key, V value) {    
    if (key == null) {        
        return putValueForNullKey(value);    
    }    

    int hash = Collections.secondaryHash(key);    
    HashMapEntry<K, V>[] tab = table;    
    int index = hash & (tab.length - 1);    
    for (HashMapEntry<K, V> e = tab[index]; e != null; e = e.next) {        
         if (e.hash == hash && key.equals(e.key)) {            
             preModify(e);            
             V oldValue = e.value;            
             e.value = value;            
             return oldValue;        
         }    
     }    

     // No entry for (non-null) key is present; create one    
     modCount++;    
     if (size++ > threshold) {        
         tab = doubleCapacity();        
         index = hash & (tab.length - 1);    
     }    
     addNewEntry(key, value, hash, index);    
     return null;
}

由源码可以看出,程序首先会检测key值是否为空,如果为空则做空值处理(null key总是存放在entryForNullKey对象中);接着对key的hashCode()做hash,然后再计算index;接着在Entry[index]下的链表里查找是否存在hash和key值与插入key的一样的HashMapEntry对象,如果有的话,则将旧值替换成value,并返回旧值;否则通过addNewEntry插入新的HashMapEntry对象,源码如下:

void addNewEntry(K key, V value, int hash, int index) {    
    table[index] = new HashMapEntry<K, V>(key, value, hash, table[index]);
}
HashMapEntry(K key, V value, int hash, HashMapEntry<K, V> next) {    
    this.key = key;    
    this.value = value;    
    this.hash = hash;    
    this.next = next;
}

由方法可知,插入方法是将新的HashMapEntry对象当成table[index]下链表的头结点,而用新的HashMapEntry对象的next指向原table[index]下链表的头结点,以达成插入链表头结点的目的。

2) get(key)
public V get(Object key) {    
    if (key == null) {        
        HashMapEntry<K, V> e = entryForNullKey;        
        return e == null ? null : e.value;    
    }    

    int hash = Collections.secondaryHash(key);    
    HashMapEntry<K, V>[] tab = table;    
    for (HashMapEntry<K, V> e = tab[hash & (tab.length - 1)];
          e != null; e = e.next) {        
          K eKey = e.key;        
          if (eKey == key || (e.hash == hash && key.equals(eKey))) {            
              return e.value;        
          }    
     }    
     return null;
}

由源码可以看出,程序首先会判断key值是否为空,如果为空,则检查entryForNullKey有没有存放值,有的话则返回;接着对key的hashCode()做hash,然后再计算index;接着在Entry[index]下的链表里查找是否存在hash和key值与插入key的一样的HashMapEntry对象,有的话则返回。

所以,根据具体来说,HashMap的存储结构是这样的:

HashMap的存储结构

三、HashMap的大小问题

  • 如果没有设置初始大小,即直接new HashMap(),size为MINIMUM_CAPACITY 的二分之一
/** * An empty table shared by all zero-capacity maps (typically from default
 * constructor). It is never written to, and replaced on first put. Its size
 * is set to half the minimum, so that the first resize will create a 
 * minimum-sized table. 
 */
private static final Entry[] EMPTY_TABLE        
      = new HashMapEntry[MINIMUM_CAPACITY >>> 1];
public HashMap() {    
      table = (HashMapEntry<K, V>[]) EMPTY_TABLE;    
      threshold = -1; // Forces first put invocation to replace EMPTY_TABLE
}
  • 如果有设置初始大小,即调用new HashMap(capacity),注意table初始大小并不是构造函数中的initialCapacity!!而是 >= initialCapacity并且是2的n次幂的整数!!!!
public HashMap(int capacity) {    
     if (capacity < 0) {        
         throw new IllegalArgumentException("Capacity: " + capacity);    
     }    

     if (capacity == 0) {        
         @SuppressWarnings("unchecked")        
         HashMapEntry<K, V>[] tab = (HashMapEntry<K, V>[]) EMPTY_TABLE;        
         table = tab;        
         threshold = -1; // Forces first put() to replace EMPTY_TABLE        
         return;    
     }    

     if (capacity < MINIMUM_CAPACITY) {        
         capacity = MINIMUM_CAPACITY;    
     } else if (capacity > MAXIMUM_CAPACITY) {        
         capacity = MAXIMUM_CAPACITY;    
     } else {        
         capacity = Collections.roundUpToPowerOfTwo(capacity);    
     }    
     makeTable(capacity);
}

其中Collections的roundUpToPowerOfTwo方法,就是获取大于等于 某个整数 并且是 2 的幂数的整数

  • 再散列rehash:当哈希表的容量超过默认容量时,doubleCapacity会调整table的为原来的2倍。这时,需要创建一张新表,将原表的映射到新表中。
private HashMapEntry<K, V>[] doubleCapacity() {    
      HashMapEntry<K, V>[] oldTable = table;    
      int oldCapacity = oldTable.length;    
      if (oldCapacity == MAXIMUM_CAPACITY) {        
          return oldTable;    
      }    
      int newCapacity = oldCapacity * 2;    
      HashMapEntry<K, V>[] newTable = makeTable(newCapacity);    
      if (size == 0) {        
          return newTable;    
      }    
      for (int j = 0; j < oldCapacity; j++) {       
           /*         
           * Rehash the bucket using the minimum number of field writes.         
           * This is the most subtle and delicate code in the class.         
           */        
           HashMapEntry<K, V> e = oldTable[j];        
           if (e == null) {            
               continue;        
           }        
           int highBit = e.hash & oldCapacity;        
           HashMapEntry<K, V> broken = null;        
           newTable[j | highBit] = e;        
           for (HashMapEntry<K, V> n = e.next; n != null; e = n, n = n.next) {            
                int nextHighBit = n.hash & oldCapacity;            
                if (nextHighBit != highBit) {                
                    if (broken == null)                    
                        newTable[j | nextHighBit] = n;                
                    else                    
                        broken.next = n;                
                        broken = e;                
                        highBit = nextHighBit;            
                 }        
           }        
           if (broken != null)            
           broken.next = null;    
       }    
       return newTable;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,951评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,606评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,601评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,478评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,565评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,587评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,590评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,337评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,785评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,096评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,273评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,935评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,578评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,199评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,440评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,163评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,133评论 2 352

推荐阅读更多精彩内容

  • 1.概述 学习本文你可以了解到: HashMap 是什么样的内部结构?有什么特点? 他的工作原理是什么样? equ...
    忽忽_阅读 389评论 0 0
  • 很多刚学Java的同学们都知道HashMap,平常一般使用,可能并不知道它的工作原理,前段时间有为刚毕业的同事在使...
    杨文杰阅读 8,281评论 6 12
  • 最近参与公司的实习生招聘工作,面试了几位实习生,我有一道每次面试都必问的题目【HasmMap的工作原理】,但很遗憾...
    竹子柳阅读 2,293评论 0 4
  • 大部分Java开发者都在使用Map,特别是HashMap。HashMap是一种简单但强大的方式去存储和获取数据。但...
    骚的掉渣阅读 791评论 0 14
  • Hash table based implementation of the Map interface. Thi...
    Johnsonxu阅读 464评论 0 0