HashMap源码分析,为什么是无序的?

疑问:HashMap是无序的,怎么做到的?

先看一个现象

Map<String, Integer> m = new HashMap<>();
for(int i=0; i<10; i++) {
    m.put("key"+i, i);
}
System.out.println(m);

结果

{key1=1, key2=2, key0=0, key5=5, key6=6, key3=3, key4=4, key9=9, key7=7, key8=8}

这说明它的内部存储结构是无序的,那它存在哪里了呢?看下put方法的源码(JDK1.7),看倒数第二行

@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,value被传到addNewEntry方法

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

它把key和value存到数组table中

transient HashMapEntry<K, V>[] table;

数组是有序的啊!在看下HashMapEntry的内部构造

static class HashMapEntry<K, V> implements Entry<K, V> {
    final K key;
    V value;
    final int hash;
    HashMapEntry<K, V> next;
    ...
}

奥,它有一个next,单向链表,这个next啥时候用的呢?在看下put方法,注意这三行的代码作用,求数据存在数组中的索引

int hash = Collections.secondaryHash(key);
HashMapEntry<K, V>[] tab = table;
int index = hash & (tab.length - 1);

hash和tab.length-1 与的结果就是的到一个tab长度范围内的一个值即求模。

    原文作者:zgengen
    原文地址: https://blog.csdn.net/zgengen/article/details/54588539
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞