Hash table
Take a look at Hash tables in wiki.
A Hash table is a data structure that accesses data stored in memory based on a Key. That is, it accesses records by computing a function on key values that maps the data for the desired query to a location in the table, which speeds up lookups. This mapping function is called a hash function, and the array of records is called a hash table.
Two key
- Construct the hash function
- To deal with conflict
HashMap
HashMap is based on the principle of hash table to implement, look at the source code to learn what data structures make up a HashMap.
-
A hash table is an array of two to the power of length.
/** * The table, resized as necessary. Length MUST Always be a power of two. */ transient HashMapEntry[] table = (HashMapEntry[]) EMPTY_TABLE; / * * * * /Copy the code
-
Now look at the HashMapEntry element that’s stored in the hash table.
static class HashMapEntry implements Map.Entry { final K key; V value; HashMapEntry next; int hash; } / * * * * /Copy the code
The included member variable HashMapEntry Next is designed as a single linked list, so the way HashMap handles conflicts is a single linked list method. See the source code for other details.
LinkedHashMap
It inherits from HashMap. Different from HashMap, the element stored in its array is LinkedHashMapEntry. In general, it is a data structure with more two-way linked lists than HashMap.
-
Bidirectional linked list header
public class LinkedHashMap extends HashMap implements Map { /** * The head of the doubly linked list. */ private transient LinkedHashMapEntry header; }Copy the code
-
The hash table holds the element LinkedHashMapEntry, inherited from HashMapEntry, with two additional member variables.
private static class LinkedHashMapEntry extends HashMapEntry { // These fields comprise the doubly linked list used for iteration. LinkedHashMapEntry before, after; }Copy the code
Before and after form a bidirectional linked list,a bidirectional cyclic linked list. This list concatenates all the elements in a hash table to form an ordered list. LinkedHashMap uses a bidirectional linked list to sort by access element or by insert element. Summary: LinkedHashMap has all the behaviors and attributes of HashMap, just a list with more behaviors and attributes. If you set sorting by access element (accessOrder == true), then every get(Object key) operation will place the element in the header header.before. Insert operations ignore the accessOrder, insert new elements at the head of the table, insert old elements (that is, the key already exists) does not change the sort order.
Deduce the diagram from the following code, existingEntry refers to the Header element, which is always used in the code: e.addbefore (Header);
private void addBefore(LinkedHashMapEntry existingEntry) {
after = existingEntry;
before = existingEntry.before;
before.after = this;
after.before = this;
}Copy the code
Bidirectional circular linked lists
If you look at the picture, where is the top of the table and where is the bottom of the table. When I say Header, I mean close to Header.
LruCache
The source code for LruCache is rarely very simple. The data structure used is a LinkedHashMap, sorted by access elements. The least recently used element is at the end of the table (eldest = header.after, header.after points to the end of the table).
public class LruCache { private final LinkedHashMap map; public LruCache(int maxSize) { if (maxSize <= 0) { throw new IllegalArgumentException("maxSize ); } this.maxSize = maxSize; This. map = new LinkedHashMap(0, 0.75f, true); } /** * Remove the eldest entries until the total of remaining entries is at or * below the requested size. * * @param maxSize the maximum size of the cache before returning. May be -1 * to evict even 0-sized elements. */ public void trimToSize(int maxSize) { while (true) { K key; V value; synchronized (this) { if (size < 0 || (map.isEmpty() && size ! = 0)) { throw new IllegalStateException(getClass().getName() + ".sizeOf() is reporting inconsistent results!" ); } if (size <= maxSize) { break; } Map.Entry toEvict = map.eldest(); if (toEvict == null) { break; } key = toEvict.getKey(); value = toEvict.getValue(); map.remove(key); size -= safeSizeOf(key, value); evictionCount++; } entryRemoved(true, key, value, null); }}Copy the code
Instead of using LruCache directly, you must override its protected int sizeOf(K key, V value) method.