`
boy00fly
  • 浏览: 197713 次
  • 性别: Icon_minigender_1
  • 来自: 南京
社区版块
存档分类
最新评论

自己动手写写:LinkedHashMap源码浅析

阅读更多

此系列文章中,上一篇是关于HashMap的源码剖析,这篇文章将向大家剖析一下LinkedHashMap的源码!

 

四. LinkedHashMap

 

我们知道从API的描述中可以看出HashMap与LinkedHashMap最大的不同在于,后者维护者一个运行于所有条目的双向链表。有了这个双向链表,就可以在迭代的时候按照插入的顺序迭代出元素(当然也可以通过LRU算法迭代元素,下面会讲到)

 

1. 类结构

public class LinkedHashMap<K, V> extends HashMap<K, V> implements Map<K, V>

 

我们可以看出LinkedHashMap继承了HashMap!

 

2. 几个重要的成员变量

/**                                                                     
 * The head of the doubly linked list.                                  
 */                                                                     
private transient Entry<K, V> header;                                   
                                                                        
/**                                                                     
 * The iteration ordering method for this linked hash map: <tt>true</tt>
 * for access-order, <tt>false</tt> for insertion-order.                
 *                                                                      
 * @serial                                                              
 */                                                                     
private final boolean accessOrder;                                      

 

 对于父类HashMap中的成员变量这里就不讲了,可参考http://boy00fly.iteye.com/blog/1139845

 其中Entry类是LinkedHashMap的内部类定义,代码片段如下:

//继承了HashMap.Entry                                       
private static class Entry<K, V> extends HashMap.Entry<K, V>
//新增的两个成员变量                                        
Entry<K, V> before, after;                                  

 

 可以看得出来新增的这两个变量就是双向链表实现的关键!!分别指向当前Entry的前一个、后一个Entry。

 

header就是指向双向链表的头部!

accessOrder:true则按照LRU算法迭代整个LinkedHashMap,false则按照元素的插入顺序迭代!

 

3. 几个重要的构造函数

   /**
     * Constructs an empty <tt>LinkedHashMap</tt> instance with the
     * specified initial capacity, load factor and ordering mode.
     *
     * @param  initialCapacity the initial capacity
     * @param  loadFactor      the load factor
     * @param  accessOrder     the ordering mode - <tt>true</tt> for
     *         access-order, <tt>false</tt> for insertion-order
     * @throws IllegalArgumentException if the initial capacity is negative
     *         or the load factor is nonpositive
     */
    public LinkedHashMap(int initialCapacity, float loadFactor, boolean accessOrder)
    {
        super(initialCapacity, loadFactor);
        this.accessOrder = accessOrder;
    }

    是不是很简单,也可参考http://boy00fly.iteye.com/blog/1139845这篇文章的构造函数分析。其中accessOrder就是我们上面所讲的控制迭代顺序的开关,只有此构造函数有这个参数,其他的构造函数默认就是false。

 

4. 几个重要的方法

   /**
     * Called by superclass constructors and pseudoconstructors (clone,
     * readObject) before any entries are inserted into the map.  Initializes
     * the chain.
     */
    void init()
    {
        header = new Entry<K, V>(-1, null, null, null);
        header.before = header.after = header;
    }

 此方法是在父类构造函数初始化的时候调用的,LinkedHashMap重写了init方法。代码中表达的意思也很明确了,这是双向链表的初始化状态。

 

 

 

/**                                                                        
 * This override alters behavior of superclass put method. It causes newly 
 * allocated entry to get inserted at the end of the linked list and       
 * removes the eldest entry if appropriate.                                
 */                                                                        
void addEntry(int hash, K key, V value, int bucketIndex)                   
{                                                                          
    createEntry(hash, key, value, bucketIndex);                            
                                                                           
    // Remove eldest entry if instructed, else grow capacity if appropriate
    Entry<K, V> eldest = header.after;                                     
    if (removeEldestEntry(eldest))                                         
    {                                                                      
        removeEntryForKey(eldest.key);                                     
    }                                                                      
    else                                                                   
    {                                                                      
        if (size >= threshold)                                             
            resize(2 * table.length);                                      
    }                                                                      
}                                                                          

/**                                                                 
 * This override differs from addEntry in that it doesn't resize the
 * table or remove the eldest entry.                                
 */                                                                 
void createEntry(int hash, K key, V value, int bucketIndex)         
{                                                                   
    HashMap.Entry<K, V> old = table[bucketIndex];                   
    Entry<K, V> e = new Entry<K, V>(hash, key, value, old);         
    table[bucketIndex] = e;                                         
    e.addBefore(header);                                            
    size++;                                                         
}                                                                   

/**                                                                   
 * Inserts this entry before the specified existing entry in the list.
 */                                                                   
private void addBefore(Entry<K, V> existingEntry)                     
{                                                                     
    after = existingEntry;                                            
    before = existingEntry.before;                                    
    before.after = this;                                              
    after.before = this;                                              
}                                                                     

 

 addEntry方法是父类HashMap的一个方法,被put相关的方法所调用即在新增元素的时候调用。

 我们通过形象的图来看一个基本的流程:

(从初始化到添加了3个元素过程中,各个元素before、after引用变化,画的有点丑,呵呵,不过意思能够表达清楚,代码的内容就不再累述了,大体和HashMap类似!)



 

 

 

再介绍一个get方法

 

/**                                                                      
 * Returns the value to which the specified key is mapped,               
 * or {@code null} if this map contains no mapping for the key.          
 *                                                                       
 * <p>More formally, if this map contains a mapping from a key           
 * {@code k} to a value {@code v} such that {@code (key==null ? k==null :
 * key.equals(k))}, then this method returns {@code v}; otherwise        
 * it returns {@code null}.  (There can be at most one such mapping.)    
 *                                                                       
 * <p>A return value of {@code null} does not <i>necessarily</i>         
 * indicate that the map contains no mapping for the key; it's also      
 * possible that the map explicitly maps the key to {@code null}.        
 * The {@link #containsKey containsKey} operation may be used to         
 * distinguish these two cases.                                          
 */                                                                      
public V get(Object key)                                                 
{                                                                        
    Entry<K, V> e = (Entry<K, V>)getEntry(key);                          
    if (e == null)                                                       
        return null;                                                     
    e.recordAccess(this);                                                
    return e.value;                                                      
}                      

/**                                                                   
  * This method is invoked by the superclass whenever the value       
  * of a pre-existing entry is read by Map.get or modified by Map.set.
  * If the enclosing Map is access-ordered, it moves the entry        
  * to the end of the list; otherwise, it does nothing.               
  */                                                                  
 void recordAccess(HashMap<K, V> m)                                   
 {                                                                    
     LinkedHashMap<K, V> lm = (LinkedHashMap<K, V>)m;                 
     if (lm.accessOrder)                                              
     {                                                                
         lm.modCount++;                                               
         remove();                                                    
         addBefore(lm.header);                                        
     }                                                                
 }         

/**                                        
 * Removes this entry from the linked list.
 */                                        
private void remove()                      
{                                          
    before.after = after;                  
    after.before = before;                 
}                                                                                                 

 

 这里我们来看一下recordAccess方法!

上面我们不是讲过accessOrder这个参数值控制着LinkedHashMap的迭代顺序嘛,这里我们来看一下。

 当accessOrder为true时,remove方法就是将当前元素从双向链表中移除,

addBefore方法再将当前元素插入到链表的头部去,这样最近读到的元素,在迭代的时候是优先被迭代出来的!

这就是所谓的LRU算法(Least Recently Used):最近最少使用算法。

当accessOrder为false时,不做任何事情,就按照插入顺序迭代出来!

 

还有些其他的方法这里也不再累述了,重点的都在上面阐述了!

  • 大小: 6 KB
  • 大小: 11.6 KB
  • 大小: 21.1 KB
  • 大小: 26.7 KB
1
4
分享到:
评论
1 楼 whg333 2013-04-25  
最后的解释recordAccess方法中说道:
addBefore方法再将当前元素插入到链表的头部去

代码中这里是addBefore(lm.header);,其实是插入到链表的尾部去吧。
正如代码注释所说的:
If the enclosing Map is access-ordered, it moves the entry to the end of the list; otherwise, it does nothing.

相关推荐

    java软件技术文档-深入java8的集合4:LinkedHashMap的实现原理.pdf

    K,V&gt; p) { } 这些方法是 HashMap 为 LinkedHashMap 提供的回调接口,用于在插入、访问和删除节点后执行特定的操作。LinkedHashMap 利用这些回调方法来维护其内部链表的顺序。 1. **afterNodeAccess(Node,V&gt; p)**: ...

    JavaLinkedHashMap源码解析Java开发Ja

    Java LinkedHashMap 是一个根据插入顺序...通过深入理解 `LinkedHashMap` 的源码,开发者可以更好地优化程序性能,尤其是在处理大量数据和需要特定顺序的场景下。同时,对于提升 Java 开发技巧和经验也是大有裨益的。

    ritelinked:LinkedHashMap和LinkedHashSet

    提供了LinkedHashMap和LinkedHashSet更多最新版本。 您可以在std或no_std环境中轻松使用它。 一些实用的功能组合,支持,帮助您更好地将其嵌入到现有代码: serde , inline-more等。特别是,它使用griddle在默认...

    Java集合系列之LinkedHashMap源码分析

    LinkedHashMap是继承自HashMap的,它重新写了一个Entry,在原来HashMap的Entry上添加了两个成员变量,分别是前继结点引用和后继结点引用。这样就将所有的结点链接在了一起,构成了一个双向链表,在获取元素的时候就...

    Java中LinkedHashMap源码解析

    Java中LinkedHashMap源码解析 LinkedHashMap是Java中的一种哈希表实现,它继承自HashMap,具有可预知的迭代顺序。LinkedHashMap维护着一个运行于所有条目的双重链表结构,该链表定义了迭代顺序,可以是插入或者访问...

    Java集合框架源码分析之LinkedHashMap详解

    Java集合框架源码分析之LinkedHashMap详解 Java集合框架中的LinkedHashMap是HashMap的子类,它继承了HashMap的存储结构,但引入了一个双向链表的头结点,将所有put到LinkedHashMap的节点连接成一个双向循环链表,...

    LinkedHashMapHelper:将LinkedHashMap转换为json,反之亦然

    将LinkedHashMap转换为json,反之亦然 如何使用 LinkedHashMap requestData = new LinkedHashMap(); LinkedHashMap auth =新的LinkedHashMap(); auth.put(“ ServiceName”,“ Login”); auth.put(“ ...

    Java LinkedHashMap工作原理及实现

    首先还是类似的,我们写一个简单的LinkedHashMap的程序:  LinkedHashMap&lt;String&gt; lmap = new LinkedHashMap();  lmap.put(语文, 1);  lmap.put(数学, 2);  lmap.put(英语, 3);  lmap.put(历史, 4); ...

    LinkedHashmap的使用

    **HashMap与LinkedHashMap的区别** HashMap是Java集合框架中的一员,它是基于哈希表实现的,提供了快速的插入、删除和查找操作,平均时间复杂度为O(1)。然而,HashMap不保证元素的顺序,迭代时元素的顺序可能与插入...

    javaSourceLearn:jdk源码构建

    List、Set、Map三大接口以及其实现类(ArrayList、LinkedList、HashSet、TreeSet、HashMap、LinkedHashMap等)的源码分析能帮助我们更高效地利用数据结构,优化程序性能。 标签“系统开源”表明这个项目鼓励开放和...

    Java集合系列(LinkedHashMap+LinkedList+ArrayList)

    Java 集合系列(LinkedHashMap+LinkedList+ArrayList) Java 集合系列是 Java 语言中的一种数据结构,用于存储和操作数据。今天,我们将介绍 Java 集合系列中的三个重要成员:LinkedHashMap、LinkedList 和 ArrayList...

    java中HashMap,LinkedHashMap,TreeMap,HashTable的区别

    ### Java中HashMap, LinkedHashMap, TreeMap,HashTable的区别 在Java编程语言中,`Map`接口是集合框架中的一个重要组成部分,用于存储键值对。本文将详细分析四种常用的`Map`实现类:`HashMap`, `LinkedHashMap`, ...

    一文搞懂Java的LinkedHashMap.docx

    《深入解析Java的LinkedHashMap》 HashMap作为Java中常用的键值对存储结构,以其高效的查找速度赢得了广大开发者们的青睐。然而,HashMap的无序性在某些特定场景下却显得力不从心,例如当我们需要按照插入顺序来...

    LinkedHashMap

    LinkedHashMap源代码,Java中Map的一种实现子类。

    set:使用LinkedHashMap在Go(Golang)中简单的Set数据结构实现

    Set是使用LinkedHashMap在Go(Golang)中简单的Set数据结构实现。 该库允许您获取一组int64或string而没有重复的项目。 用法 package main import ( "fmt" "github.com/StudioSol/set" ) func main () { ...

Global site tag (gtag.js) - Google Analytics