1 LinkedHashMap (before jdk1.7)
We know that the underlying data store of Map is a hash table (array + one-way linked list). Next, let's take a look at another LinkedHashMap, which is a subclass of HashMap. It maintains a two-way linked list (Hash Table + two-way linked list) on the basis of HashMap. The insertion order (first in first out, similar to FIFO) or the least recently used (LRU) order can be used during traversal.
Let's take a look at the implementation of LinkedHashMap.
1.1 definitions
public class LinkedHashMap<K,V> extends HashMap<K,V> implements Map<K,V>
From the definition, we can see that LinkedHashMap inherits from HashMap and implements the Map interface. This means that some excellent factors of HashMap can be inherited, such as hash addressing, fast search using linked lists to solve hash conflicts, and whether LinkedHashMap has been optimized for some inefficient contents in HashMap, such as container expansion process and traversal mode. Keep looking at the code.
1.2 underlying storage
LinkedHashMap is based on HashMap and maintains a two-way linked list. That is to say, LinkedHashMap is the implementation of a hash table (array + one-way linked list) + two-way linked list. What is the implementation method
/** * 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. * * @serial11 */ private final boolean accessOrder;
I saw a very familiar attribute header, which appeared in the LinkedList. The English annotation is very clear. It is the head node of the two-way linked list, right.
Let's look at the accessOrder attribute. true indicates the order used less recently, and false indicates the insertion order. Of course, why don't you see the array? Don't forget that LinkedHashMap inherits from HashMap
Let's look at the difference between the node class of Entry and that in HashMap.
/** * LinkedHashMap entry. */ private static class Entry<K,V> extends HashMap.Entry<K,V> { // These fields comprise the doubly linked list used for iteration. // The previous node before and the next node after of the bidirectional linked list Entry<K,V> before, after ; // The constructor directly calls the constructor (super) of the parent class HashMap Entry( int hash, K key, V value, HashMap.Entry<K,V> next) { super(hash, key, value, next); } /** * Method of deleting current node from linked list */ private void remove() { // Change the reference relationship between the two nodes before and after the current node. After the current node is not referenced, gc can recycle // Point the after of the previous node to the next node before.after = after; // Point the before of the next node to the previous node after.before = before; } /** * Add a node before the specified node to the linked list (that is, to the end of the linked list) */ private void addBefore(Entry<K,V> existingEntry) { // Now change your direction to the front and back // Point the after of the current node to the given node (add it to the front of existingEntry) after = existingEntry; // Point the before of the current node to the previous node of the given node 33 before = existingentry.before; // Now change your own direction before and after // The after of the previous node points to itself before.after = this; // The before of the next few points to yourself after.before = this; } // This method is called when the query element or modification element (put the same key) is obtained from the Map void recordAccess(HashMap<K,V> m) { LinkedHashMap<K,V> lm = (LinkedHashMap<K,V>)m; // If accessOrder is true, that is, the least recently used order is used if (lm.accessOrder ) { lm. modCount++; // Delete first and then add, which is equivalent to moving // Delete current element remove(); // Add the current element before the header (that is, the end of the linked list) addBefore(lm. header); } } // Call this method when deleting elements from the Map void recordRemoval(HashMap<K,V> m) { remove(); } }
You can see that the Entry inherits the Entry in the HashMap, but the Entry in the LinkedHashMap has two more attributes, which point to the before of the previous node and the after of the next node. It is these two attributes that form a two-way linked list. The Entry also has an inherited next attribute, which is used to point to the next node in the one-way linked list. What's the matter, Why is it a one-way linked list and a two-way linked list? In fact, you are right. The node here is a node in the one-way linked list in the Hash table. It is also a node in the two-way linked list maintained by LinkedHashMap. Do you feel tall for a moment
Note: the black arrow indicates the next direction of the one-way linked list, the red arrow indicates the before direction of the two-way linked list, and the blue arrow indicates the after direction of the two-way linked list. In addition, LinkedHashMap also has a header node that does not save data, which is not drawn here.
As can be seen from the above figure, LinkedHashMap is still a hash table. The bottom layer is composed of an array, and each item of the array is a one-way linked list, pointing to the next node from next. However, the difference between LinkedHashMap is that there are two more attributes before and after in the node. These two attributes form a two-way circular linked list, which maintains the order of elements in the Map container. Look at the recordRemoval method in Entry. This method will be called when the node is deleted. After the normal deletion of the linked list node in the Hash table, the method is called to correct the backward pointing relation of the bidirectional linked list after the node is deleted. From this point of view, LinkedHashMap is slower than add, remove and set of HashMap, because it needs to maintain two way linked list.
1.3 construction method
/** * Construct a LinkedHashMap specifying the initial capacity and loading factor. The default accessOrder is false */ public LinkedHashMap( int initialCapacity, float loadFactor) { super(initialCapacity, loadFactor); accessOrder = false; } /** * Construct a LinkedHashMap with a specified initial capacity. The default accessOrder is false */ public LinkedHashMap( int initialCapacity) { super(initialCapacity); accessOrder = false; } /** * Construct a LinkedHashMap using the default initial capacity (16) and the default loading factor (0.75). The default accessOrder is false */ public LinkedHashMap() { super(); accessOrder = false; } /** * Construct a LinkedHashMap of the specified map. The created LinkedHashMap uses the default loading factor (0.75) and the initial capacity sufficient to accommodate the specified map. The default accessOrder is false. */ public LinkedHashMap(Map<? extends K, ? extends V> m) { super(m); accessOrder = false; } /** * Construct a LinkedHashMap that specifies the initial capacity, load factor, and accessOrder */ public LinkedHashMap( int initialCapacity, float loadFactor, boolean accessOrder) { super(initialCapacity, loadFactor); 40 this.accessOrder = accessOrder; 41 }
The construction method is very simple. It basically calls the construction method (super) of the parent HashMap. The only difference is the setting of accessOrder. Most of the above construction parameters set accessOrder to false by default. Only one construction method leaves an exit to set the accessOrder parameter. After reading the construction method, I found a problem. Where is the initialization of the header node
Recall the construction method of HashMap:
/** * Constructs an empty <tt>HashMap</tt> with the specified initial * capacity and load factor. * * @param initialCapacity the initial capacity * @param loadFactor the load factor * @throws IllegalArgumentException if the initial capacity is negative * or the load factor is nonpositive */ public HashMap( int initialCapacity, float loadFactor) { if (initialCapacity < 0) throw new IllegalArgumentException( "Illegal initial capacity: " + initialCapacity); if (initialCapacity > MAXIMUM_CAPACITY) initialCapacity = MAXIMUM_CAPACITY; if (loadFactor <= 0 || Float.isNaN(loadFactor)) throw new IllegalArgumentException( "Illegal load factor: " + loadFactor); // Find a power of 2 >= initialCapacity int capacity = 1; while (capacity < initialCapacity) capacity <<= 1; this.loadFactor = loadFactor; threshold = (int)(capacity * loadFactor); table = new Entry[capacity]; init(); } /** * Initialization hook for subclasses. This method is called * in all constructors and pseudo -constructors (clone, readObject) * after HashMap has been initialized but before any entries have * been inserted. (In the absence of this method, readObject would * require explicit knowledge of subclasses.) */ void init() {
init() is an empty method in HashMap, that is, a callback function reserved for subclasses. Let's take a look at the implementation of init() method in LinkedHashMap
/** * Called by superclass constructors and pseudoconstructors (clone, * readObject) before any entries are inserted into the map. Initializes * the chain. */ void init() { // Initialize the session header, set hash to - 1, and set key, value, and next to null header = new Entry<K,V>(-1, null, null, null); // Both before and after of the header point to the header itself header.before = header. after = header ;
1.4 add
LinkedHashMap does not override the put method, but overrides the addEntry called by the put method in the HashMap
/** * 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) { // Call the createEntry method to create a new node createEntry(hash, key, value, bucketIndex); // Remove eldest entry if instructed, else grow capacity if appropriate // Take out the first node after the header (because the header does not save data, take the first node after the header) Entry<K,V> eldest = header.after ; // Judge whether the capacity is insufficient, delete the first node or expand the capacity if (removeEldestEntry(eldest)) { // Delete the first node (Cache that can implement FIFO and LRU policies) removeEntryForKey(eldest. key); } else { // Capacity expansion is the same as HashMap 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) { // The logic of the following three lines of code is to create a new node and put it at the head of the one-way linked list // Fetch the old node at the bucket index position of the array HashMap.Entry<K,V> old = table[bucketIndex]; // Create a new node and point next to the old node Entry<K,V> e = new Entry<K,V>(hash, key, value, old); // Put the newly created node in the bucket index position of the array table[bucketIndex] = e; // Maintain the two-way linked list and add new nodes in front of the two-way linked list header (at the end of the list) e.addBefore( header); // Counter size plus 1 size++; } /** * false is returned by default, that is, elements will not be deleted. If you want to implement the cache function, you only need to override this method */ protected boolean removeEldestEntry(Map.Entry<K,V> eldest) { return false; }
It can be seen that there are two more logic in the addition method than in HashMap. One is to judge whether to delete the first element or expand the capacity when the Map capacity is insufficient, and the other is to maintain a two-way linked list. When judging whether to delete elements, we found that the removeEldestEntry method always returns false. Originally, if you want to implement the Cache function, you need to inherit the LinkedHashMap and rewrite the removeEldestEntry method. The container function is provided by default.
1.5 deletion
LinkedHashMap does not override the remove method, but implements the recordRemoval method of the Entry class. This method is a callback method provided by HashMap. Callback is carried out in the remove method of HashMap. Of course, the main function of recordRemoval in LinkedHashMap is to maintain a two-way linked list
1.6 search
LinkedHashMap rewrites the get method, but does reuse the getEntry method in the HashMap. LinkedHashMap refers to adding the logic to call the recoreAccess method in the get method. Of course, the purpose of the recoreAccess method is to maintain a two-way linked list. Return to the above logic to see the recoreAccess method of the Entry class
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 ; }
1.7 does it include
/** * Returns <tt>true</tt> if this map maps one or more keys to the * specified value. * * @param value value whose presence in this map is to be tested * @return <tt> true</tt> if this map maps one or more keys to the * specified value */ public boolean containsValue(Object value) { // Overridden to take advantage of faster iterator // Traverse the two-way linked list to find the specified value if (value==null) { for (Entry e = header .after; e != header; e = e.after ) if (e.value ==null) return true; } else { for (Entry e = header .after; e != header; e = e.after ) if (value.equals(e.value )) return true; } return false; }
LinkedHashMap rewrites containsValue. containsValue of HashMap needs to traverse the entire hash table, which is very inefficient. After rewriting in LinkedHashMap, instead of traversing the hash table, it traverses the two-way linked list maintained by it. Does this improve the efficiency? Let's analyze the following: the hash table is composed of array + one-way linked list. Due to the use of hash algorithm, the hash may be uneven, and even some items of the array have no elements (no hash gives the corresponding hash value). However, there are no empty items in the two-way linked list of LinkedHashMap, so the containsValue efficiency of LinkedHashMap is better than that of HashMap.
1.8 cache function
In the end, let's simply implement a Cache function based on LInkedHashMap
import java.util.LinkedHashMap; import java.util.Map; public class MyLocalCache extends LinkedHashMap<String, Object> { private static final long serialVersionUID = 7182816356402068265L; private static final int DEFAULT_MAX_CAPACITY = 1024; private static final float DEFAULT_LOAD_FACTOR = 0.75f; private int maxCapacity; public enum Policy { FIFO, LRU } public MyLocalCache(Policy policy) { super(DEFAULT_MAX_CAPACITY, DEFAULT_LOAD_FACTOR, Policy.LRU .equals(policy)); this.maxCapacity = DEFAULT_MAX_CAPACITY; } public MyLocalCache(int capacity, Policy policy) { super(capacity, DEFAULT_LOAD_FACTOR, Policy. LRU.equals(policy)); this.maxCapacity = capacity; } @Override protected boolean removeEldestEntry(Map.Entry<String, Object> eldest) { return this.size() > maxCapacity; } public static void main(String[] args) { MyLocalCache cache = new MyLocalCache(5, Policy.LRU); cache.put( "k1", "v1" ); cache.put( "k2", "v2" ); cache.put( "k3", "v3" ); cache.put( "k4", "v4" ); cache.put( "k5", "v5" ); cache.put( "k6", "v6" ); System. out.println("size=" + cache.size()); System. out.println("----------------------" ); for (Map.Entry<String, Object> entry : cache.entrySet()) { System. out.println(entry.getValue()); } System. out.println("----------------------" ); System. out.println("k3=" + cache.get("k3")); System. out.println("----------------------" ); for (Map.Entry<String, Object> entry : cache.entrySet()) { System. out.println(entry.getValue()); } } }