码迷,mamicode.com
首页 > 其他好文 > 详细

OpenJDK1.8.0 源码解析————HashMap的实现(二)

时间:2016-06-05 17:08:54      阅读:213      评论:0      收藏:0      [点我收藏+]

标签:

    刚才简单介绍了HashMap的一部分的知识,算是为下面HashMap的进一步学习做准备吧。

    刚才一直在思考的一个问题是,这方面的知识网上的资料也是一抓一大把,即使是这样我为什么还要花费时间去写呢。后来我仔细想了一下,其实很简单,虽然大家解读的是同一份源码,但是如果只是看看别人写的文章,源码它真正的思想和魅力你都体会不到一半。所以还是决定自己写写,虽然和别人写的大同小异,但是写完真的能体会到更深层次的东西。再就是我的描述或许不准确甚至说是有错误。希望看到的人可以指出,这样对我也是一种帮助。

    然后觉得有点扯远了,赶紧回到正题吧。

    下面要说的就是HashMap的具体操作了。这里我主要说的是put方法和get方法,以及这两个方法中包含的其他方法。我讲到的地方算是我理解到的(有的可能不准确)。还有一些没有讲到,是因为个人能力有限,没有理解透彻,所以就不误导大家了。
    

    首先我们先看put方法。

 public V put(K key, V value) {
      return putVal(hash(key), key, value, false, true);
 }

 


    调用put方法之后调用了hash(key)方法,我们先看一下这个hash()方法,这个hash方法就是定位一个hashmap的位置。

    static final int hash(Object key) {
      int h;
      return (key == null) ? 0 : (h = key.hashCode()) ^ (h >>> 16);
    }

 

    

    这个过程的含义就是:


      如果要put进table的Key值为null,返回0;如果Key值不为null,返回 key的hashCode值 和 key的hashCode无符号算术右移16位的值 按位异或的结果

      算术右移的规则是给右移后高位补0

      按位异或就是把两个数转化为按二进制,按位进行比较,每一位上的数相同就取0,不同就取1

      按照算术右移的规则,正数在算术右移之后会变小;负数在算术右移后会变成正数

      因此一个正数右移16位换句话说就是丢弃低16为位。那么对于任何小于2的16次方的数,右移16后结果都为0

      例如:2的16次方为65536转化为二进制为:1 0000 0000 0000 0000,右移16为刚好为0

      小于2的16次方的数,例如:2的10次方为1024转化为二进制位:10000000000,右移16为0

      当一个数右移为0时,它和任何数 按位异或 结果都是这个数本身

      所以这个hash()函数对于非null的key,仅在key的hashCode值大于等于2的16次方的时候才会重新调整其值。

      其他时候hash函数返回值就是key的hashCode。

      然后为什么设计成为这样呢?我们可以看到put方法里除了调用到了hash()方法外,调用到了putVal方法。putVal方法如下:

      

    final V putVal(int hash, K key, V value, boolean onlyIfAbsent, boolean evict) {

        Node<K,V>[] tab; Node<K,V> p; int n, i;
        if ((tab = table) == null || (n = tab.length) == 0)
        n = (tab = resize()).length;
        if ((p = tab[i = (n - 1) & hash]) == null)
        tab[i] = newNode(hash, key, value, null);
        ......
      }

        在这里我们可以看到有这样的代码段

        tab[i = (n - 1) & hash]

         tab[i] = newNode(hash, key, value, null)

         实际上就是先给 i 赋值为 (n - 1) & hash。n就是table的长度,假设默认为16,然后hash值的得出就是上面所讲的

         然后给tab[i]装入一个新的Node。此处的关键就是i的取值

         我们默认n为16,然后16-1=15,转化为二进制后为 1111 

           此时,假设hash的值不做 h = key.hashCode() ^ (h >>> 16) 这样的处理

           而是直接调用 h = key.hashCode() 得到h,那么1111和h按位与或,i 结果就为0

         那么传入的键值对放的位置就是tab[0]位置或者说是在挂tab[0]位置下面的链表的一个节点

             这样的话大多数类似于 xxxx xxxx 0000这样的数 和 1111 按位与或结果都为0

           那么tab[0]位置有可能会存储很多的值,即链表的长度会很长,这样查找时就会降低了性能。

           现在我们看看hash值经过 h = key.hashCode() ^ (h >>> 16) 这样的处理后是怎样的结果

           做一个简单的测试

public static void testHash(){
    System.out.println(Integer.toBinaryString("aaaa".hashCode()) + "=>"+Integer.toBinaryString("aaaa".hashCode() ^ ("aaaa".hashCode() >>>16)));
    System.out.println(Integer.toBinaryString("bbbb".hashCode()) + "=>"+Integer.toBinaryString("bbbb".hashCode() ^ ("bbbb".hashCode() >>>16)));
    System.out.println(Integer.toBinaryString("aabb".hashCode()) + "=>"+Integer.toBinaryString("aabb".hashCode() ^ ("aabb".hashCode() >>>16)));
    System.out.println(Integer.toBinaryString("cccc".hashCode()) + "=>"+Integer.toBinaryString("cccc".hashCode() ^ ("cccc".hashCode() >>>16)));
    System.out.println(Integer.toBinaryString("dddd".hashCode()) + "=>"+Integer.toBinaryString("dddd".hashCode() ^ ("dddd".hashCode() >>>16)));
}

        调用这个方法结果为下图

        

        技术分享

        "=>"前面的是没经过 h = key.hashCode()) ^ (h >>> 16)处理之后得到的哈希码的二进制表示形式

        "=>"后面的是经过了 h = key.hashCode()) ^ (h >>> 16)处理之后得到的哈希码的二进制表示形式

        可以看到没经过处理直接拿到的值后面的四位都为0000,这样和 n-1 按位与或后结果都为0

        这样把键值为aaaa,bbbb,cccc,dddd的对象放入Map中最终都放到了挂到了tab[0]的后面

        而经过处理后拿到的值的二进制表示后面的四位都是不一样的,这样和 n-1 按位与或后结果就全为0,也就是在hashCode()的基础在做了散列

        至于为什么要右移16位,看到大多数人的说法就是折中(因为容量定义为int类型,4个字节,32位,右移16算是折中做法)

        

        接着我们继续说put方法,put方法的实现是通过调用putVal方法。

        putVal完整的源码如下:

        

final V putVal(int hash, K key, V value, boolean onlyIfAbsent,
                   boolean evict) {
                
                Node<K,V>[] tab; Node<K,V> p; int n, i;
                //如果table为空或者table的长度为0,证明table尚未创建(第一次调用put时会发生这种情况),此时创建table
                if ((tab = table) == null || (n = tab.length) == 0)
                    n = (tab = resize()).length;
                /*
                    走到这一步说明table不为空。
                    判断下标为i的tab是否存在结点,没有则创建新节点  
                */
                if ((p = tab[i = (n - 1) & hash]) == null)
                    tab[i] = newNode(hash, key, value, null);
                else {
                    /*
                        走到这里说明tab[i]存在节点
                        就意味着发生了冲突,这时就要处理冲突
                    */
                    Node<K,V> e; K k;
                    
                    /*
                        此时的 p = tab[i = (n - 1) & hash] ,是上一步if操作产生的结果
                        如果存在于tab中节点tab[i]与传入节点的key和value相等,则记录下当前存在于tab中的tab[i]节点    
                    */
                    if (p.hash == hash &&
                        ((k = p.key) == key || (key != null && key.equals(k))))
                        e = p;
                    /*
                        如果当前tab[i]节点类型为红黑树,则按照红黑树方法添加传入的元素
                    */
                    else if (p instanceof TreeNode)
                        e = ((TreeNode<K,V>)p).putTreeVal(this, tab, hash, key, value);
                    else {
                    /*
                        走到这一步说明当前首tab[i]结点类型为链表类型。
                        然后循环遍历链表
                    */
                        for (int binCount = 0; ; ++binCount) {
                            // 如果遍历到末尾时,先在尾部追加该元素结点。
                            if ((e = p.next) == null) {
                                p.next = newNode(hash, key, value, null);
                                //追加完成后,如果对应tab[i]下的节点大于8,则把tab[i]节点下的链表结构转化为红黑树结构
                                if (binCount >= TREEIFY_THRESHOLD - 1) // -1 for 1st
                                    treeifyBin(tab, hash);
                                break;
                            }
                            /*
                                判断所有遍历到的节点,如果key和value都和传入的key和value相等,则停止for循环遍历
                                此时该节点在上一个if操作中的e = p.next中已经记录了下来
                            */
                            if (e.hash == hash &&
                                ((k = e.key) == key || (key != null && key.equals(k))))
                                break;
                                
                            //让p节点指向下一个节点.
                            p = e;
                        }
                    }
                    /*
                        如果e不为空,证明传入的key和value在哈希表中有相同元素的结点
                        则用传入的value替换e.value,并且返回e.value
                    */
                    
                    if (e != null) { // existing mapping for key
                        V oldValue = e.value;
                        if (!onlyIfAbsent || oldValue == null)
                            e.value = value;
                        //一个没有任何实现的空方法
                        afterNodeAccess(e);
                        return oldValue;
                    }
                }
                /*
                    到此一个节点处理完毕,共做了三次判断
                    在总结一下:
                    ①初次调用put方法,table为空,创建一个table,返回table的长度
                    ②table不为空后,判断tab[i]是否存在节点,不存在创建节点并赋值给tab[i];如果存在节点,走第三步
                    ③解决冲突。分别三步。这里就不写了,可以回头看。
                    完成这三不步,一个节点处理完毕,tab发生一次修改,因此modCount++    
                */
                ++modCount;
                //如果存放元素的个数大于HashMap的阈值,则进行扩容
                if (++size > threshold)
                    resize();
                //一个没有任何实现的空实现
                afterNodeInsertion(evict);
                return null;
            }

        

      到此put方法结束。

      但实际上还没有结束,因为put方法涉及到的扩容机制还没有讨论。

      接着谈一下扩容机制的实现,也就是resize方法,

      该方法可用来初始化HashMap大小,也可以重新调整HashMap大小变为原来2倍大小

      完整的源码如下:

    

 final Node<K,V>[] resize() {
                    /*
                        把table放到oldTab,table可能为空,可能不为空
                        为空就是对table进行初始化
                        不为空就是对table进行扩容
                    */
                    Node<K,V>[] oldTab = table;
//如果旧的哈希表oldTable为空则旧的哈希表容量oldCap为0;不为空容量oldCap就为oldTab的长度 int oldCap = (oldTab == null) ? 0 : oldTab.length;
//给旧的填充因子oldThr赋值默认的填充因子0.75 /oldThr不会是0,因为有tableSizeFor()方法,threshold至少是1,这个在前面做过一个小测试。 int oldThr = threshold;
//定义将要初始化或者扩充的哈希表的容量和填充因子 int newCap, newThr = 0; //满足这个if条件证明进行的是扩容操作 if (oldCap > 0) { /* 如果将要进行扩容的tab的容量大于HashMap的最大的容量,表明不能在进行扩容了 此时进行的操作是把int能表示的最大值赋值给hashMap的阈值,然后返回之前本就存在的tab */ if (oldCap >= MAXIMUM_CAPACITY) { threshold = Integer.MAX_VALUE; return oldTab; } else if ((newCap = oldCap << 1) < MAXIMUM_CAPACITY && oldCap >= DEFAULT_INITIAL_CAPACITY) /* 进入这里表明可以进行扩容 新的容量为之前容量的二倍。具体代码就是newCap = oldCap << 1 新的阈值为原来的二倍。 具体的代码 newThr = oldThr << 1 */ newThr = oldThr << 1; } //oldCap=0 ,oldThr>0,tab为空,因此oldCap为0,而oldThr=threshold > 0 else if (oldThr > 0) //以 new HashMap(int initialCapacity) 方式或者 HashMap(int initialCapacity, float loadFactor) 创建的一个HashMap //有初始容量,有阈值,有加载因子。但是依然是一个空的HashMap //因此这样的两种创建方式调用put会到这个分支 //设置新的tab容量为之前的阈值 //然后在下面创建一个newCap大小的桶数组,即执行Node<K,V>[] newTab = (Node<K,V>[])new Node[newCap]; newCap = oldThr; else { //以 new HashMap()方式创建一个HashMap, //只有一个加载因子,HashMap为空,因此这种方式创建的HashMap对象在第一次调用put时会进入到这个分支 //oldCap=0,oldThr=0 ,进行table的初始化操作,即使用默认填充比和初始容量对table进行初始化 //设置新的hash的桶数组的长度newCap为默认值16 newCap = DEFAULT_INITIAL_CAPACITY; //newThr = 0.75×16 = 12,当size值大于12时,进行扩容 newThr = (int)(DEFAULT_LOAD_FACTOR * DEFAULT_INITIAL_CAPACITY); } //根据负载因子设置极限值 if (newThr == 0) { float ft = (float)newCap * loadFactor; newThr = (newCap < MAXIMUM_CAPACITY && ft < (float)MAXIMUM_CAPACITY ? (int)ft : Integer.MAX_VALUE); } threshold = newThr; @SuppressWarnings({"rawtypes","unchecked"}) Node<K,V>[] newTab = (Node<K,V>[])new Node[newCap]; table = newTab; //这个新的HashMap newTab创建好之后,如果之前的HashMap不为空,就要把之前的oldTab导入到新的newTab,最后返回这个新的newTab if (oldTab != null) { for (int j = 0; j < oldCap; ++j) { Node<K,V> e; if ((e = oldTab[j]) != null) { oldTab[j] = null; if (e.next == null) newTab[e.hash & (newCap - 1)] = e; else if (e instanceof TreeNode) ((TreeNode<K,V>)e).split(this, newTab, j, oldCap); else { // preserve order Node<K,V> loHead = null, loTail = null; Node<K,V> hiHead = null, hiTail = null; Node<K,V> next; do { next = e.next; if ((e.hash & oldCap) == 0) { if (loTail == null) loHead = e; else loTail.next = e; loTail = e; } else { if (hiTail == null) hiHead = e; else hiTail.next = e; hiTail = e; } } while ((e = next) != null); if (loTail != null) { loTail.next = null; newTab[j] = loHead; } if (hiTail != null) { hiTail.next = null; newTab[j + oldCap] = hiHead; } } } } } return newTab; }

     

      重要的put方法讲完了之后,接着说get方法。

      get方法是根据键获取相应的值。

      源码如下:

        public V get(Object key) {
                    Node<K,V> e;
                      //实际上是根据传入键的哈希值去哈希表里找对应的节点的值
                      return (e = getNode(hash(key), key)) == null ? null : e.value;
             }
                
             final Node<K,V> getNode(int hash, Object key) {
                    Node<K,V>[] tab; Node<K,V> first, e; int n; K k;
                    //确保HashMap里维护的table不为空,且传入的键值转化为索引后,也就是定位到了table数组的下标,
                    //该下标对应的table节点不为空,进行后续操作;否则直接返回空
                    if ((tab = table) != null && (n = tab.length) > 0 &&
                        (first = tab[(n - 1) & hash]) != null) {
//判断找到的节点的hash和key是否和传入的hash和key相等,如果相等直接返回这个节点 if (first.hash == hash && // always check first node ((k = first.key) == key || (key != null && key.equals(k)))) return first;
//走这一步说明该hash值在对应到了table数组,但是该位置所存储的节点的键值和传入的键值不等。 //说明hash值有冲突的,因此要向下遍历链表或者红黑树 if ((e = first.next) != null) { //如果是红黑树,遍历红黑树找到相应节点并返回 if (first instanceof TreeNode) return ((TreeNode<K,V>)first).getTreeNode(hash, key); do { //走到这一步说明是链表,则遍历链表找到相应节点并返回 if (e.hash == hash && ((k = e.key) == key || (key != null && key.equals(k)))) return e; } while ((e = e.next) != null); } } //如果从这一步返回说明没找到,返回null return null; }

         

          到此我要讲的就完了

          漏掉的部分有链表转红黑树,还有就是如何把之前的旧table导入到扩容之后的新table

          其实主要难懂的操作就是红黑树,链表的操作大家应该很容易就看懂了

          所以后面关于红黑树的知识还要再看看

          如果看懂了我就会继续做出总结

      

 

OpenJDK1.8.0 源码解析————HashMap的实现(二)

标签:

原文地址:http://www.cnblogs.com/HouJiao/p/5561062.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!