`

Java集合HashMap的实现原理(借鉴)

 
阅读更多

HashMap 和 HashSet 是 Java Collection Framework 的两个重要成员,其中 HashMap 是 Map 接口的常用实现类,HashSet 是 Set 接口的常用实现类。虽然 HashMap 和 HashSet 实现的接口规范不同,但它们底层的 Hash 存储机制完全一样,甚至 HashSet 本身就采用 HashMap 来实现的。 

通过 HashMapHashSet 的源代码分析其 Hash 存储机制

        实际上,HashSet 和 HashMap 之间有很多相似之处,对于 HashSet 而言,系统采用 Hash 算法决定集合元素的存储位置,这样可以保证能快速存、取集合元素;对于 HashMap 而言,系统 key-value 当成一个整体进行处理,系统总是根据 Hash 算法来计算 key-value 的存储位置,这样可以保证能快速存、取 Map 的 key-value 对。 在介绍集合存储之前需要指出一点:虽然集合号称存储的是 Java 对象,但实际上并不会真正将 Java 对象放入 Set 集合中,只是在 Set 集合中保留这些对象的引用而言。也就是说:Java 集合实际上是多个引用变量所组成的集合,这些引用变量指向实际的 Java 对象。 集合和引用 就像引用类型的数组一样,当我们把 Java 对象放入数组之时,并不是真正的把 Java 对象放入数组中,只是把对象的引用放入数组中,每个数组元素都是一个引用变量。 

如何存放多个键值对,下面通过一个简单的实例看看:

HashMap<String , Double> map = new HashMap<String , Double>();    
map.put("语文" , 80.0);    
map.put("数学" , 89.0);    
map.put("英语" , 78.2);   

看看HashMap中相关方法put(作用是:如果所插入的key在HashTable表中以前存在,则用最新的value代替原先的value,but原先的key不变)的调用:

复制代码
/**
     * Associates the specified value with the specified key in this map.
     * If the map previously contained a mapping for the key, the old
     * value is replaced.

     * @param key key with which the specified value is to be associated
     * @param value value to be associated with the specified key
     * @return the previous value associated with key, or
     *         null if there was no mapping for key.
     * (A null return can also indicate that the map previously associated   null with key.)
     */
    public V put(K key, V value) {
        if (key == null)
            return putForNullKey(value);
        int hash = hash(key);
        //搜索指定hash值在对应table中的索引.
        int i = indexFor(hash, table.length);
        //如果i处索引处的Entry不为null,则通过循环不断遍历e元素的下一个元素.
        for (Entry<K,V> e = table[i]; e != null; e = e.next) {
            Object k;
            if (e.hash == hash && ((k = e.key) == key || key.equals(k))) {
                V oldValue = e.value;
                e.value = value;
                e.recordAccess(this);
                return oldValue;
            }
        }

        modCount++;
        addEntry(hash, key, value, i);
        return null;
    }
复制代码
    /**
     * Returns index for hash code h.
     */
    static int indexFor(int h, int length) {
        return h & (length-1);
    }

 

       上述indexFor这个方法非常巧妙,它总是通过 h &(table.length -1) 来得到该对象的保存位置(二进制相与——而 HashMap 底层数组的长度总是 的 n 次方

当 length 总是 的倍数时,h & (length-1) 将是一个非常巧妙的设计:假设 h=5,length=16, 那么 h & length - 1 将得到 5;如果 h=6,length=16, 那么 h & length - 1 将得到 6 ……如果 h=15,length=16, 那么 h & length - 1 将得到 15;但是当 h=16 时 , length=16 时,那么 h & length - 1 将得到 了;当 h=17 时 , length=16 时,那么 h & length - 1 将得到 ……这样保证计算得到的索引值总是位于 table 数组的索引之内。 

 

       根据上面 put 方法的源代码可以看出,当程序试图将一个 key-value 对放入 HashMap 中时,程序首先根据该 key 的 hashCode() 返回值决定该 Entry 的存储位置:如果两个 Entry 的 key 的 hashCode() 返回值相同,那它们的存储位置相同。如果这两个 Entry 的 key 通过 equals 比较返回 true,新添加 Entry 的 value 将覆盖集合中原有 Entry 的 value,但 key 不会覆盖。如果这两个 Entry 的 key 通过 equals 比较返回 false,新添加的 Entry 将与集合中原有 Entry 形成 Entry 链,而且新添加的 Entry 位于 Entry 链的头部——具体说明继续看 addEntry() 方法的说明。 当向 HashMap 中添加 key-value 对,由其 key 的 hashCode() 返回值决定该 key-value 对(就是 Entry 对象)的存储位置。当两个 Entry 对象的 key 的 hashCode() 返回值相同时,将由 key 通过 equals() 比较值决定是采用覆盖行为(返回 true),还是产生 Entry 链(返回 false)。 上面程序中还调用了 addEntry(hash, key, value, i); 代码,其中 addEntry 是 HashMap 提供的一个包访问权限的方法,该方法仅用于添加一个 key-value 对。下面是该方法的代码: 

 

 

复制代码
void addEntry(int hash, K key, V value, int bucketIndex)    
{    
    // 获取指定 bucketIndex 索引处的 Entry    
    Entry<K,V> e = table[bucketIndex];     // ①   
    // 将新创建的 Entry 放入 bucketIndex 索引处,并让新的 Entry 指向原来的 Entry    
    table[bucketIndex] = new Entry<K,V>(hash, key, value, e);    
    // 如果 Map 中的 key-value 对的数量超过了极限   
    if (size++ >= threshold)    
        // 把 table 对象的长度扩充到 2 倍。   
        resize(2 * table.length);    // ②   
}   
复制代码

 

       上面addEntry方法的代码很简单,但其中包含了一个非常优雅的设计:系统总是将新添加的 Entry 对象放入 table 数组的 bucketIndex 索引处——如果 bucketIndex 索引处已经有了一个 Entry 对象,那新添加的 Entry 对象指向原有的 Entry 对象(产生一个 Entry 链),如果 bucketIndex 索引处没有 Entry 对象,也就是上面程序号代码的 e 变量是 null,也就是新放入的 Entry 对象指向 null,也就是没有产生 Entry 链,如下图所示:

 

        当 HashMap 的每个 bucket 里存储的 Entry 只是单个 Entry ——也就是没有通过指针产生 Entry 链时,此时的 HashMap 具有最好的性能:当程序通过 key 取出对应 value 时,系统只要先计算出该 key 的 hashCode() 返回值,在根据该 hashCode 返回值找出该 key 在 table 数组中的索引,然后取出该索引处的 Entry,最后返回该 key 对应的 value 即可。看 HashMap 类的 get(Object key) 方法代码:

 

复制代码
public V get(Object key) {
        if (key == null)
            return getForNullKey();
        Entry<K,V> entry = getEntry(key);

        return null == entry ? null : entry.getValue();
    }

    /**
     * Returns the entry associated with the specified key in the
     * HashMap.  Returns null if the HashMap contains no mapping
     * for the key.
     */
    final Entry<K,V> getEntry(Object key) {
        int hash = (key == null) ? 0 : hash(key);
        for (Entry<K,V> e = table[indexFor(hash, table.length)];
             e != null;
             e = e.next) {
            Object k;
            if (e.hash == hash &&
                ((k = e.key) == key || (key != null && key.equals(k))))
                return e;
        }
        return null;
    }
复制代码

       归纳起来简单地说,HashMap 在底层将 key-value 当成一个整体进行处理,这个整体就是一个 Entry 对象。HashMap 底层采用一个 Entry[] 数组来保存所有的 key-value 对,当需要存储一个 Entry 对象时,会根据 Hash 算法来决定其存储位置;当需要取出一个 Entry 时,也会根据 Hash 算法找到其存储位置,直接取出该 Entry

      

       当创建 HashMap 时,有一个默认的负载因子load factor),其默认值为 0.75,这是时间和空间成本上一种折衷:增大负载因子可以减少 Hash 表(就是那个 Entry 数组)所占用的内存空间,但会增加查询数据的时间开销,而查询是最频繁的的操作(HashMap 的 get() 与 put() 方法都要用到查询);减小负载因子会提高数据查询的性能,但会增加 Hash 表所占用的内存空间。

 掌握了上面知识之后,我们可以在创建 HashMap 时根据实际需要适当地调整 load factor 的值;如果程序比较关心空间开销、内存比较紧张,可以适当地增加负载因子;如果程序比较关心时间开销,内存比较宽裕则可以适当的减少负载因子。通常情况下,程序员无需改变负载因子的值。 如果开始就知道 HashMap 会保存多个 key-value 对,可以在创建时就使用较大的初始化容量,如果 HashMap 中 Entry 的数量一直不会超过极限容量(capacity * load factor),HashMap 就无需调用 resize() 方法重新分配 table 数组,从而保证较好的性能。当然,开始就将初始容量设置太高可能会浪费空间(系统需要创建一个长度为 capacity 的 Entry 数组),因此创建 HashMap 时初始化容量设置也需要小心对待。 

分享到:
评论

相关推荐

    JAVA集合类应用[借鉴].pdf

    Java集合类是Java编程语言中用于存储和管理对象的核心组件,位于`java.util`包中。集合类的主要优点在于它们的灵活性,能够适应各种数据结构需求,而不像数组那样需要预先知道存储对象的数量。集合类存放的是对象的...

    Java集合排序及java集合类详解(Collection、List、Map、Set)借鉴.pdf

    Java集合框架是Java编程语言中的核心部分,它提供了一种高效、灵活的数据组织和操作机制。本文将深入探讨集合框架的各个组成部分,包括Collection、List、Set和Map,以及它们的特点、常用方法和实现原理。 1. 集合...

    JAVA程序源代码学习借鉴

    6. **集合框架**:Java集合框架提供了许多接口和类,如ArrayList、LinkedList、HashMap等,用于存储和操作数据集合。 7. **输入/输出(I/O)**:Java的I/O流用于读写文件、网络通信等,学习InputStream和...

    Java源码角度分析HashMap用法

    Java源码角度分析HashMap用法主要介绍了Java源码角度分析HashMap用法,具有一定借鉴价值,需要的朋友可以参考下。下面我们将从HashMap的优点、缺点、使用方法、源码分析等方面进行深入分析。 HashMap的优点 ...

    Java面试题和答案借鉴.pdf

    Java是一种广泛使用的面向对象的编程语言,其设计目标之一就是具有自动内存管理功能,通过垃圾收集器(GC)来自动回收不再使用的对象所占用的内存空间。GC的存在使得程序员无需手动进行内存分配和释放,降低了内存...

    Java编程的逻辑

    接下来,深入到Java的实现原理。Java程序首先通过编译器转换为字节码(.class文件),然后在Java虚拟机(JVM)上运行。JVM是Java平台的核心,它负责加载、验证、执行字节码,提供内存管理和垃圾回收服务。理解JVM的...

    JAVA基础考试题借鉴.pdf

    3. 集合框架:`Collection`是Java集合框架的顶级接口,`ArrayList`和`HashSet`都是其子类。A选项中的`TreeMap`属于`Map`接口而非`Collection`,B选项中的`Hashtable`也是`Map`接口的实现。C和D选项正确。 4. 线程:...

    JAVA各种计算工具类[借鉴].pdf

    - `HashMap`, `HashSet`, `TreeMap`: 这些都是Java集合框架中的数据结构,用于存储和操作键值对(HashMap和TreeMap)或无序唯一元素(HashSet)。 - `Iterator`: 迭代器接口,用于遍历集合中的元素。 9. **第三方...

    Java面向对象程序设计教学大纲[借鉴].pdf

    Java集合框架是Java API中用于存储和操作对象集合的一个体系结构。它包括各种接口和实现类,例如List、Set、Map等。 考虑到提供的文件内容存在重复和混乱,上述知识点是基于常规Java面向对象程序设计教学大纲的可能...

    学习Java必须读懂两套源代码

    例如,集合框架中的ArrayList和LinkedList是如何实现动态扩容的,HashMap的散列策略如何避免碰撞,以及线程同步的底层机制如synchronized关键字的工作原理等。 其次,理解JVM的源代码,包括类加载、字节码解析、...

    Java面试之笑傲江湖

    2. **Java面试之独孤九剑**:可能对应的是Java集合框架的掌握,包括ArrayList、LinkedList、HashMap、HashSet等数据结构。理解它们的工作原理、性能特点和适用场景,能灵活运用解决实际问题,是展示编程功底的重要...

    Java中的区别[借鉴].pdf

    在Java编程语言中,了解和掌握这些基础知识至关重要。让我们详细探讨一下给定文件中提到的知识点: 1. **作用域**: - **public**: 可以被任何类访问。 - **private**: 只能在定义该成员的类内部访问。 - **...

    Java 基础核心总结.pdf

    Java是一种广泛使用的编程语言,它由Sun Microsystems公司在1995年发布,设计上借鉴了C和C++的语言特性,并增加了面向对象的编程特点。Java语言具有跨平台性,能够一次编写,到处运行,这是因为Java编写的应用程序被...

    JAVA夜未眠(java学习笔记).rar

    6. **集合框架**:List、Set、Map接口及其实现类(如ArrayList、LinkedList、HashSet、HashMap等)是Java编程中经常用到的数据结构,学习笔记会详细解释其原理和使用方法。 7. **输入输出流**:Java提供了丰富的IO...

    java7源码-collection-comments:Java7集合框架源码注释

    首先,让我们关注Java集合框架的主要组件。集合框架主要包括接口和实现这些接口的类,它们可以分为两大类:List和Set。List接口代表有序的集合,允许有重复元素,如ArrayList和LinkedList。Set接口则表示不包含重复...

    java性能优化借鉴

    ### Java性能优化借鉴 #### 一、合理使用单例模式 单例模式是软件开发中常用的模式之一,它确保一个类只有一个实例,并提供一个全局访问点。在Java中使用单例模式可以有效控制资源的使用,减少不必要的实例化操作...

    Java通用范例 开发金典

    4. **集合框架**:Java集合框架包括List、Set、Map等接口及其实现类,如ArrayList、LinkedList、HashSet、HashMap等,这部分代码可以帮助理解如何存储和操作数据。 5. **IO流**:Java的IO流系统支持输入输出操作,...

    Java底层知识点、源码解读,技术栈相关原理知识点、工具解读最佳实践、功能点实战,问题排查,开发技巧等

    9. **集合框架**: 遍历HashMap、ArrayList、LinkedList、TreeSet等集合的实现原理,掌握它们在不同场景下的选择和优化。 10. **网络编程**: Java的Socket编程、HTTP客户端库(如HttpURLConnection、HttpClient)...

    疯狂Java讲义源码(第二部分)

    在第5章“Java集合框架”中,主要讲解了Java提供的各种数据结构和容器类,如ArrayList、LinkedList、HashSet、HashMap等。这些集合类是Java程序员日常工作中不可或缺的部分。ArrayList和LinkedList分别基于动态数组...

    北大计算机系JAVA培训讲义

    此外,Java集合框架是另一个重要的主题,包括ArrayList、LinkedList、HashMap等常用数据结构的使用和实现原理。这些集合类提供了高效的数据存储和操作,是实际编程中不可或缺的工具。 多线程编程也是Java的一大特色...

Global site tag (gtag.js) - Google Analytics