zoukankan      html  css  js  c++  java
  • OpenJDK1.8.0 源码解析————HashMap的实现(二)

        上一篇文章介绍了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

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

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

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

          

  • 相关阅读:
    SQL CAST与CONVERT区别
    SQL存储过程相关信息查看
    SQLServer系统变量使用
    转 C#中哈希表(HashTable)的用法详解
    SQL中的全局变量和局部变量(@@/@)
    SqlServer中用SQL语句附加数据库及修改数据库逻辑文件名
    SQL Server中常用全局变量介绍
    SQL Server 用户定义表类型
    03- 手机App功能测试要点以及登录页面的测试
    1. APP移动端性能测试基础知识入门
  • 原文地址:https://www.cnblogs.com/HouJiao/p/5561062.html
Copyright © 2011-2022 走看看