Java容器(List、Set、Map)知识点快速复习手册(中)

Java容器(List、Set、Map)知识点快速复习手册(中)

前言

本文快速回顾了Java中容器的知识点,用做面试复习,事半功倍。html

上篇:主要为容器概览,容器中用到的设计模式,List源码java

中篇:Map源码git

下篇:Set源码,容器总结github

其它知识点复习手册

  • Java基础知识点面试手册(上)
  • Java基础知识点面试手册(下)
  • Java容器(List、Set、Map)知识点快速复习手册(上)

HashMap

http://wiki.jikexueyuan.com/project/java-collection/hashmap.html面试

源码分析:算法

http://www.javashuo.com/article/p-ykttmamr-gg.html编程

关键词

  • 初始容量16
  • 扩容是2倍,加载因子0.75
  • 头插法
  • 0桶存放null
  • 从 JDK 1.8 开始,一个桶存储的链表长度大于 8 时会将链表转换为红黑树(前提:键值对要超过64个)
  • 自动地将传入的容量转换为2的幂次方
  • 保证运算速度:确保用位运算代替模运算来计算桶下标。hash& (length-1)运算等价于对 length 取模。
  • hash均匀分布:数据在数组上分布就比较均匀,而且可以利用所有二进制位,也就是说碰撞的概率小,
  • table数组+Entry
    []链表(散列表),红黑树
  • 扩容操做须要把键值对从新插入新的 table 中,从新计算全部key有特殊机制(JDK1.8后)

存储结构

hashMap的一个内部类Node:segmentfault

1static class Node<K,V> implements Map.Entry<K,V> {
2        final int hash;
3        final K key;
4        V value;
5        Node<K,V> next; //链表结构,存储下一个元素

Java容器(List、Set、Map)知识点快速复习手册(中)
在这里插入图片描述
Node内部包含了一个 Entry 类型的数组table,数组中的每一个位置被当成一个桶。设计模式

1transient Entry[] table;

Entry 存储着键值对。它包含了四个字段,从 next 字段咱们能够看出 Entry 是一个链表。即数组中的每一个位置被当成一个桶,一个桶存放一个链表。数组

HashMap 使用拉链法来解决冲突,同一个链表中存放哈希值相同的 Entry。

1static class Entry<K,V> implements Map.Entry<K,V> {
 2    final K key;
 3    V value;
 4    Entry<K,V> next;
 5    int hash;
 6
 7    Entry(int h, K k, V v, Entry<K,V> n) {
 8        value = v;
 9        next = n;
10        key = k;
11        hash = h;
12    }
13
14    public final K getKey() {
15        return key;
16    }
17
18    public final V getValue() {
19        return value;
20    }
21
22    public final V setValue(V newValue) {
23        V oldValue = value;
24        value = newValue;
25        return oldValue;
26    }
27
28    public final boolean equals(Object o) {
29        if (!(o instanceof Map.Entry))
30            return false;
31        Map.Entry e = (Map.Entry)o;
32        Object k1 = getKey();
33        Object k2 = e.getKey();
34        if (k1 == k2 || (k1 != null && k1.equals(k2))) {
35            Object v1 = getValue();
36            Object v2 = e.getValue();
37            if (v1 == v2 || (v1 != null && v1.equals(v2)))
38                return true;
39        }
40        return false;
41    }
42
43    public final int hashCode() {
44        return Objects.hashCode(getKey()) ^ Objects.hashCode(getValue());
45    }
46
47    public final String toString() {
48        return getKey() + "=" + getValue();
49    }
50}

构造器

Java容器(List、Set、Map)知识点快速复习手册(中)
在这里插入图片描述

构造时就会调用tableSizeFor():返回一个大于输入参数且最近的2的整数次幂。

1static final int tableSizeFor(int cap) {
2    int n = cap - 1;
3    n |= n >>> 1;
4    n |= n >>> 2;
5    n |= n >>> 4;
6    n |= n >>> 8;
7    n |= n >>> 16;
8    return (n < 0) ? 1 : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY : n + 1;
9}

拉链法

应该注意到链表的插入是以头插法方式进行的

1HashMap<String, String> map = new HashMap<>();
2map.put("K1", "V1");
3map.put("K2", "V2");
4map.put("K3", "V3");
  • 新建一个 HashMap,默认大小为 16;
  • 插入 <K1,V1> 键值对,先计算 K1 的 hashCode 为 115,使用除留余数法获得所在的桶下标 115%16=3。
  • 插入 <K2,V2> 键值对,先计算 K2 的 hashCode 为 118,使用除留余数法获得所在的桶下标 118%16=6。
  • 插入 <K3,V3> 键值对,先计算 K3 的 hashCode 为 118,使用除留余数法获得所在的桶下标 118%16=6,插在 <K2,V2> 前面。

查找须要分红两步进行:

计算键值对所在的桶;
在链表上顺序查找,时间复杂度显然和链表的长度成正比。

put 操做

  • 当咱们 put 的时候,若是 key 存在了,那么新的 value 会代替旧的 value
  • 若是 key 存在的状况下,该方法返回的是旧的 value,
  • 若是 key 不存在,那么返回 null。
1public V put(K key, V value) {
 2    if (table == EMPTY_TABLE) {
 3        inflateTable(threshold);
 4    }
 5    // 键为 null 单独处理
 6    if (key == null)
 7        return putForNullKey(value);
 8    int hash = hash(key);
 9    // 肯定桶下标
10    int i = indexFor(hash, table.length);
11    // 先找出是否已经存在键为 key 的键值对,若是存在的话就更新这个键值对的值为 value
12    for (Entry<K,V> e = table[i]; e != null; e = e.next) {
13        Object k;
14        if (e.hash == hash && ((k = e.key) == key || key.equals(k))) {
15            V oldValue = e.value;
16            e.value = value;
17            e.recordAccess(this);
18            return oldValue;
19        }
20    }
21
22    modCount++;
23    // 插入新键值对
24    addEntry(hash, key, value, i);
25    return null;
26}

HashMap 容许插入键为 null 的键值对。可是由于没法调用 null 的 hashCode() 方法,也就没法肯定该键值对的桶下标,只能经过强制指定一个桶下标来存放。HashMap 使用第 0 个桶存放键为 null 的键值对。

1private V putForNullKey(V value) {
 2    for (Entry<K,V> e = table[0]; e != null; e = e.next) {
 3        if (e.key == null) {
 4            V oldValue = e.value;
 5            e.value = value;
 6            e.recordAccess(this);
 7            return oldValue;
 8        }
 9    }
10    modCount++;
11    addEntry(0, null, value, 0);
12    return null;
13}

使用链表的头插法,也就是新的键值对插在链表的头部,而不是链表的尾部。

1void addEntry(int hash, K key, V value, int bucketIndex) {
 2    if ((size >= threshold) && (null != table[bucketIndex])) {
 3        resize(2 * table.length);
 4        hash = (null != key) ? hash(key) : 0;
 5        bucketIndex = indexFor(hash, table.length);
 6    }
 7
 8    createEntry(hash, key, value, bucketIndex);
 9}
10
11void createEntry(int hash, K key, V value, int bucketIndex) {
12    Entry<K,V> e = table[bucketIndex];
13    // 头插法,链表头部指向新的键值对
14    table[bucketIndex] = new Entry<>(hash, key, value, e);
15    size++;
16}
1Entry(int h, K k, V v, Entry<K,V> n) {
2    value = v;
3    next = n;
4    key = k;
5    hash = h;
6}

补充:hashmap里hash方法的高位优化:

http://www.javashuo.com/article/p-wcboirew-cq.html

https://note.youdao.com/yws/res/18743/50AADC7BB42845B29CDA293FC409250C?ynotemdtimestamp=1548155508277

设计者将key的哈希值的高位也作了运算(与高16位作异或运算,使得在作&运算时,此时的低位其实是高位与低位的结合),这就增长了随机性,减小了碰撞冲突的可能性!

为什么要这么作?

table的长度都是2的幂,所以index仅与hash值的低n位有关,hash值的高位都被与操做置为0了。

这样作很容易产生碰撞。设计者权衡了speed, utility, and quality,将高16位与低16位异或来减小这种影响。设计者考虑到如今的hashCode分布的已经很不错了,并且当发生较大碰撞时也用树形存储下降了冲突。仅仅异或一下,既减小了系统的开销,也不会形成的由于高位没有参与下标的计算(table长度比较小时),从而引发的碰撞。

肯定桶下标

不少操做都须要先肯定一个键值对所在的桶下标。

1int hash = hash(key);
2int i = indexFor(hash, table.length);

4.1 计算 hash 值

1final int hash(Object k) {
 2    int h = hashSeed;
 3    if (0 != h && k instanceof String) {
 4        return sun.misc.Hashing.stringHash32((String) k);
 5    }
 6
 7    h ^= k.hashCode();
 8
 9    // This function ensures that hashCodes that differ only by
10    // constant multiples at each bit position have a bounded
11    // number of collisions (approximately 8 at default load factor).
12    h ^= (h >>> 20) ^ (h >>> 12);
13    return h ^ (h >>> 7) ^ (h >>> 4);
14}
1public final int hashCode() {
2    return Objects.hashCode(key) ^ Objects.hashCode(value);
3}

4.2 取模

令 x = 1<\<\4,即 \x 为 2 的 4 次方,它具备如下性质:

1x   : 00010000
2x-1 : 00001111

令一个数 y 与 x-1 作与运算,能够去除 y 位级表示的第 4 位以上数:

1y       : 10110010
2x-1     : 00001111
3y&(x-1) : 00000010

这个性质和 y 对 x 取模效果是同样的:

1y   : 10110010
2x   : 00010000
3y%x : 00000010

咱们知道,位运算的代价比求模运算小的多,所以在进行这种计算时用位运算的话能带来更高的性能。

肯定桶下标的最后一步是将 key 的 hash 值对桶个数取模:hash%capacity,若是能保证 capacity 为 2 的 n 次方,那么就能够将这个操做转换为位运算。

1static int indexFor(int h, int length) {
2    return h & (length-1);
3}

当 length 老是 2 的 n 次方时,h& (length-1)运算等价于对 length 取模,也就是 h%length,可是 & 比 % 具备更高的效率。这看上去很简单,其实比较有玄机的,咱们举个例子来讲明:

<table>

  • 从上面的例子中能够看出:当它们和 15-1(1110)“与”的时候,8 和 9产生了相同的结果,也就是说它们会定位到数组中的同一个位置上去,这就产生了碰撞,8 和 9 会被放到数组中的同一个位置上造成链表,那么查询的时候就须要遍历这个链 表,获得8或者9,这样就下降了查询的效率。

  • 同时,咱们也能够发现,当数组长度为 15 的时候,hash 值会与 15-1(1110)进行“与”,那么最后一位永远是 0,而 0001,0011,0101,1001,1011,0111,1101 这几个位置永远都不能存放元素了,空间浪费至关大,数组可使用的位置比数组长度小了不少,这意味着进一步增长了碰撞的概率。

  • 而当数组长度为16时,即为2的n次方时,2n-1 获得的二进制数的每一个位上的值都为 1,这使得在低位上&时,获得的和原 hash 的低位相同,加之 hash(int h)方法对 key 的 hashCode 的进一步优化,加入了高位计算,就使得只有相同的 hash 值的两个值才会被放到数组中的同一个位置上造成链表。

因此说,当数组长度为 2 的 n 次幂的时候,不一样的 key 算得得 index 相同的概率较小,那么数据在数组上分布就比较均匀,也就是说碰撞的概率小

扩容-基本原理

设 HashMap 的 table 长度为 M,须要存储的键值对数量为 N,若是哈希函数知足均匀性的要求,那么每条链表的长度大约为 N/M,所以平均查找次数的复杂度为 O(N/M)。

为了让查找的成本下降,应该尽量使得 N/M 尽量小,所以须要保证 M 尽量大,也就是说 table 要尽量大。HashMap 采用动态扩容来根据当前的 N 值来调整 M 值,使得空间效率和时间效率都能获得保证。

和扩容相关的参数主要有:capacity、size、threshold 和 load_factor。
Java容器(List、Set、Map)知识点快速复习手册(中)

1static final int DEFAULT_INITIAL_CAPACITY = 16;
 2
 3static final int MAXIMUM_CAPACITY = 1 << 30;
 4
 5static final float DEFAULT_LOAD_FACTOR = 0.75f;
 6
 7transient Entry[] table;
 8
 9transient int size;
10
11int threshold;
12
13final float loadFactor;
14
15transient int modCount;

从下面的添加元素代码中能够看出,当须要扩容时,令 capacity 为原来的两倍。

1void addEntry(int hash, K key, V value, int bucketIndex) {
2    Entry<K,V> e = table[bucketIndex];
3    table[bucketIndex] = new Entry<>(hash, key, value, e);
4    if (size++ >= threshold)
5        resize(2 * table.length);
6}

扩容使用 resize() 实现,须要注意的是,扩容操做一样须要把 oldTable 的全部键值对从新插入 newTable 中,所以这一步是很费时的。

1void resize(int newCapacity) {
 2    Entry[] oldTable = table;
 3    int oldCapacity = oldTable.length;
 4    if (oldCapacity == MAXIMUM_CAPACITY) {
 5        threshold = Integer.MAX_VALUE;
 6        return;
 7    }
 8    Entry[] newTable = new Entry[newCapacity];
 9    transfer(newTable);
10    table = newTable;
11    threshold = (int)(newCapacity * loadFactor);
12}
13
14void transfer(Entry[] newTable) {
15    Entry[] src = table;
16    int newCapacity = newTable.length;
17    for (int j = 0; j < src.length; j++) {
18        Entry<K,V> e = src[j];
19        if (e != null) {
20            src[j] = null;
21            do {
22                Entry<K,V> next = e.next;
23                int i = indexFor(e.hash, newCapacity);
24                e.next = newTable[i];
25                newTable[i] = e;
26                e = next;
27            } while (e != null);
28        }
29    }
30}

扩容-从新计算桶下标

Rehash优化:http://www.javashuo.com/article/p-edvgtahx-gy.html

在进行扩容时,须要把键值对从新放到对应的桶上。HashMap 使用了一个特殊的机制,能够下降从新计算桶下标的操做。

假设原数组长度 capacity 为 16,扩容以后 new capacity 为 32:

1capacity     : 00010000
2new capacity : 00100000

对于一个 Key,

  • 它的哈希值若是在第 5 位上为 0,那么取模获得的结果和以前同样;
  • 若是为 1,那么获得的结果为原来的结果 +16。

总结:

通过rehash以后,元素的位置要么是在原位置,要么是在原位置再移动2次幂的位置

所以,咱们在扩充HashMap的时候,不须要像JDK1.7的实现那样从新计算hash,只须要看看原来的hash值新增的那个bit是1仍是0就行了,是0的话索引没变,是1的话索引变成“原索引+oldCap”,能够看看下图为16扩充为32的resize示意图:
Java容器(List、Set、Map)知识点快速复习手册(中)
在这里插入图片描述

计算数组容量

HashMap 构造函数容许用户传入的容量不是 2 的 n 次方,由于它能够自动地将传入的容量转换为 2 的 n 次方。

先考虑如何求一个数的掩码,对于 10010000,它的掩码为 11111111,可使用如下方法获得:

1mask |= mask >> 1    11011000
2mask |= mask >> 2    11111110
3mask |= mask >> 4    11111111
4

mask+1 是大于原始数字的最小的 2 的 n 次方。

1num     10010000
2mask+1 100000000

如下是 HashMap 中计算数组容量的代码:

1static final int tableSizeFor(int cap) {
2    int n = cap - 1;
3    n |= n >>> 1;
4    n |= n >>> 2;
5    n |= n >>> 4;
6    n |= n >>> 8;
7    n |= n >>> 16;
8    return (n < 0) ? 1 : (n >= MAXIMUM_CAPACITY) ? MAXIMUM_CAPACITY : n + 1;
9}

链表转红黑树

并非桶子上有8位元素的时候它就能变成红黑树,它得同时知足咱们的键值对大于64才行的

这是为了不在哈希表创建初期,多个键值对刚好被放入了同一个链表中而致使没必要要的转化。

HashTable

关键词:

  • Hashtable的迭代器不是 fail-fast,HashMap 的迭代器是 fail-fast 迭代器。
  • Hashtable 的 key 和 value 都不容许为 null,HashMap 能够插入键为 null 的 Entry。
  • HashTable 使用 synchronized 来进行同步。
  • 基于 Dictionary 类(遗留类)
  • HashMap 不能保证随着时间的推移 Map 中的元素次序是不变的。

HashMap 与 HashTable

Java容器(List、Set、Map)知识点快速复习手册(中)
在这里插入图片描述

  • HashTable 基于 Dictionary 类(遗留类),而 HashMap 是基于 AbstractMap。
  • Dictionary 是任何可将键映射到相应值的类的抽象父类,
  • 而AbstractMap是基于Map接口的实现,它以最大限度地减小实现此接口所需的工做。
  • HashMap 的 key 和 value 都容许为 null,而 Hashtable 的 key 和 value 都不容许为 null
  • HashMap 的迭代器是 fail-fast 迭代器,而 Hashtable 的 enumerator 迭代器不是 fail-fast 的。
  • 因为 Hashtable 是线程安全的也是 synchronized,因此在单线程环境下它比 HashMap 要慢。
  • Hashtable 中的几乎全部的 public 的方法都是synchronized的,而有些方法也是在内部经过 synchronized 代码块来实现。
  • 可是在 Collections 类中存在一个静态方法:synchronizedMap(),该方法建立了一个线程安全的 Map 对象,并把它做为一个封装的对象来返回。
  • 也可使用 ConcurrentHashMap,它是 HashTable 的替代,并且比 HashTable 可扩展性更好

ConcurrentHashMap

谈谈ConcurrentHashMap1.7和1.8的不一样实现:

http://www.importnew.com/23610.html

详细源码分析(还未细看):

https://blog.csdn.net/yan_wenliang/article/details/51029372

http://www.javashuo.com/article/p-zggcwnug-bu.html

主要针对jdk1.7的实现来介绍

关键词

  • key和value都不容许为null
  • Hashtable是将全部的方法进行同步,效率低下。而ConcurrentHashMap经过部分锁定+CAS算法来进行实现线程安全的
  • get方法是非阻塞,无锁的。重写Node类,经过volatile修饰next来实现每次获取都是最新设置的值
  • 在高并发环境下,统计数据(计算size…等等)实际上是无心义的,由于在下一时刻size值就变化了。
  • 实现形式不一样:
  • 1.7:Segment + HashEntry的方式进行实现
  • 1.8:与HashMap相同(散列表(数组+链表)+红黑树)采用Node数组 + CAS + Synchronized来保证并发安全进行实现

存储结构

jdk1.7

jdk1.7中采用Segment + HashEntry的方式进行实现
Java容器(List、Set、Map)知识点快速复习手册(中)
在这里插入图片描述

Segment:其继承于 ReentrantLock 类,从而使得 Segment 对象能够充当锁的角色。

Segment 中包含HashBucket的数组,其能够守护其包含的若干个桶。

1static final class HashEntry<K,V> {
2    final int hash;
3    final K key;
4    volatile V value;
5    volatile HashEntry<K,V> next;
6}

ConcurrentHashMap采用了分段锁,每一个分段锁维护着几个桶,多个线程能够同时访问不一样分段锁上的桶,从而使其并发度更高(并发度就是 Segment 的个数)。

jdk1.8
Java容器(List、Set、Map)知识点快速复习手册(中)

在这里插入图片描述

  • JDK 1.7 使用分段锁机制来实现并发更新操做,核心类为 Segment,它继承自重入锁 ReentrantLock,并发程度与 Segment 数量相等。

  • JDK 1.8 使用了 CAS 操做来支持更高的并发度,在 CAS 操做失败时使用内置锁 synchronized。

  • 而且 JDK 1.8 的实现也在链表过长时会转换为红黑树。

1.8中放弃了Segment臃肿的设计,取而代之的是采用Node数组 + CAS + Synchronized来保证并发安全进行实现

添加元素:put

Java容器(List、Set、Map)知识点快速复习手册(中)
在这里插入图片描述

只让一个线程对散列表进行初始化!

获取元素:get

从顶部注释咱们能够读到,get方法是不用加锁的,是非阻塞的。

Node节点是重写的,设置了volatile关键字修饰,导致它每次获取的都是最新设置的值

获取大小:size

每一个 Segment 维护了一个 count 变量来统计该 Segment 中的键值对个数。

在执行 size 操做时,须要遍历全部 Segment 而后把 count 累计起来。

ConcurrentHashMap 在执行 size操做时先尝试不加锁,若是连续两次不加锁操做获得的结果一致,那么能够认为这个结果是正确的。

尝试次数使用 RETRIES_BEFORE_LOCK 定义,该值为 2,retries 初始值为 -1,所以尝试次数为 3。

若是尝试的次数超过 3 次,就须要对每一个 Segment 加锁。

删除元素:remove

Java容器(List、Set、Map)知识点快速复习手册(中)
在这里插入图片描述

为何用这么方式删除呢,细心的同窗会发现上面定义的HashEntry的key和next都是final类型的,因此不能改变next的指向,因此又复制了一份指向删除的结点的next。

Collections.synchronizedMap()与ConcurrentHashMap的区别

参考:https://blog.csdn.net/lanxiangru/article/details/53495854

  • Collections.synchronizedMap()和Hashtable同样,实现上在调用map全部方法时,都对整个map进行同步,而ConcurrentHashMap的实现却更加精细,它对map中的全部桶加了锁,同步操做精确控制到桶,因此,即便在遍历map时,其余线程试图对map进行数据修改,也不会抛出ConcurrentModificationException。
  • ConcurrentHashMap从类的命名就能看出,它是个HashMap。而Collections.synchronizedMap()能够接收任意Map实例,实现Map的同步。好比TreeMap。

总结

ConcurrentHashMap 的高并发性主要来自于三个方面:

  • 用分离锁实现多个线程间的更深层次的共享访问。
  • 用 HashEntery对象的不变性来下降执行读操做的线程在遍历链表期间对加锁的需求。
  • 经过对同一个 Volatile 变量的写 / 读访问,协调不一样线程间读 / 写操做的内存可见性。

LinkedHashMap

http://wiki.jikexueyuan.com/project/java-collection/linkedhashmap.html

http://www.javashuo.com/article/p-fscfnemy-ct.html

关键词

  • 容许使用 null 值和 null 键
  • 此实现不是同步的(linkedlist,lilnkedhashset也不是同步的)
  • 维护着一个运行于全部条目的双向链表。定义了迭代顺序,该迭代顺序能够是插入顺序或者是访问顺序。
  • 初始容量对遍历没有影响:遍历的双向链表,而不是散列表
  • 在访问顺序的状况下,使用get方法也是结构性的修改(会致使Fail-Fast)

概论

Java容器(List、Set、Map)知识点快速复习手册(中)
在这里插入图片描述
Java容器(List、Set、Map)知识点快速复习手册(中)
在这里插入图片描述

成员变量

该 Entry 除了保存当前对象的引用外,还保存了其上一个元素 before 和下一个元素 after的引用,从而在哈希表的基础上又构成了双向连接列表。

1/**
 2* LinkedHashMap的Entry元素。
 3* 继承HashMap的Entry元素,又保存了其上一个元素before和下一个元素after的引用。
 4 */
 5static class Entry<K,V> extends HashMap.Node<K,V> {
 6        Entry<K,V> before, after;
 7        Entry(int hash, K key, V value, Node<K,V> next) {
 8            super(hash, key, value, next);
 9        }
10    }

构造器

Java容器(List、Set、Map)知识点快速复习手册(中)
在这里插入图片描述

  • 经过源代码能够看出,在 LinkedHashMap 的构造方法中,实际调用了父类 HashMap 的相关构造方法来构造一个底层存放的 table 数组,但额外能够增长 accessOrder 这个参数,若是不设置

  • 默认为 false,表明按照插入顺序进行迭代;
  • 固然能够显式设置为 true,表明以访问顺序进行迭代。
  • 在构建新节点时,构建的是LinkedHashMap.Entry 再也不是Node.

获取元素:get

LinkedHashMap 重写了父类 HashMap 的 get 方法,实际在调用父类 getEntry() 方法取得查找的元素后,再判断当排序模式 accessOrder 为 true 时,记录访问顺序,将最新访问的元素添加到双向链表的表头,并从原来的位置删除。

因为的链表的增长、删除操做是常量级的,故并不会带来性能的损失。

遍历元素

为啥注释说:初始容量对遍历没有影响?

由于它遍历的是LinkedHashMap内部维护的一个双向链表,而不是散列表(固然了,链表双向链表的元素都来源于散列表)

LinkedHashMap应用

http://wiki.jikexueyuan.com/project/java-collection/linkedhashmap-lrucache.html

LRU最近最少使用(访问顺序)

用这个类有两大好处:

  • 它自己已经实现了按照访问顺序或插入顺序的存储
  • LinkedHashMap 自己有removeEldestEntry方法用于判断是否须要移除最不常读取的数,可是,原始方法默认不须要移除,咱们须要override这样一个方法。

Java里面实现LRU缓存一般有两种选择:

  • 使用LinkedHashMap
  • 本身设计数据结构,使用链表+HashMap

如下是使用 LinkedHashMap 实现的一个 LRU 缓存:

  • 设定最大缓存空间 MAX_ENTRIES 为 3;
  • 使用 LinkedHashMap 的构造函数将 accessOrder 设置为 true,开启 LRU 顺序;
  • 覆盖 removeEldestEntry() 方法实现,在节点多于 MAX_ENTRIES 就会将最近最久未使用的数据移除。
1class LRUCache<K, V> extends LinkedHashMap<K, V> {
 2    private static final int MAX_ENTRIES = 3;
 3
 4    protected boolean removeEldestEntry(Map.Entry eldest) {
 5        return size() > MAX_ENTRIES;
 6    }
 7
 8    LRUCache() {
 9        super(MAX_ENTRIES, 0.75f, true);
10    }
11}
1public static void main(String[] args) {
2    LRUCache<Integer, String> cache = new LRUCache<>();
3    cache.put(1, "a");
4    cache.put(2, "b");
5    cache.put(3, "c");
6    cache.get(1);
7    cache.put(4, "d");
8    System.out.println(cache.keySet());
9}
1[3, 1, 4]

实现详细代码请参考文章:补充知识点-缓存

FIFO(插入顺序)

还能够在插入顺序的LinkedHashMap直接重写下removeEldestEntry方法便可轻松实现一个FIFO缓存

TreeMap

关键词

  • 红黑树
  • 非同步
  • key不能为null
  • 实现了NavigableMap接口,而NavigableMap接口继承着SortedMap接口,是有序的(HahMap是Key无序的)
  • TreeMap的基本操做 containsKey、get、put 和 remove 的时间复杂度是 log(n) 。
  • 适用于查找性能要求不那么高,反而对有序性要求比较高的应用场景
  • 使用Comparator或者Comparable来比较key是否相等与排序的问题

概览

Java容器(List、Set、Map)知识点快速复习手册(中)
在这里插入图片描述

获取元素:get

详细看:

http://www.javashuo.com/article/p-wumkjbiq-ng.html

总结:

  • 若是在构造方法中传递了Comparator对象,那么就会以Comparator对象的方法进行比较。不然,则使用Comparable的compareTo(T o)方法来比较。
  • 值得说明的是:若是使用的是compareTo(T o)方法来比较,key必定是不能为null,而且得实现了Comparable接口的。
  • 即便是传入了Comparator对象,不用compareTo(T o)方法来比较,key也是不能为null的

删除元素:remove

  • 删除节点而且平衡红黑树

参考

  • https://github.com/CyC2018/CS-Notes/blob/master/docs/notes/Java%20%E5%AE%B9%E5%99%A8.md
  • 公众号:Java3y
  • Eckel B. Java 编程思想 [M]. 机械工业出版社, 2002.
  • Java Collection Framework
  • Iterator 模式
  • Java 8 系列之从新认识 HashMap
  • What is difference between HashMap and Hashtable in Java?
  • Java 集合之 HashMap
  • The principle of ConcurrentHashMap analysis
  • 探索 ConcurrentHashMap 高并发性的实现机制
  • HashMap 相关面试题及其解答
  • Java 集合细节(二):asList 的缺陷
  • Java Collection Framework – The LinkedList Class

关注我

本人目前为后台开发工程师,主要关注Python爬虫,后台开发等相关技术。

原创博客主要内容

  • 笔试面试复习知识点手册
  • Leetcode算法题解析(前150题)
  • 剑指offer算法题解析
  • Python爬虫相关实战
  • 后台开发相关实战

同步更新如下博客

Csdn

http://blog.csdn.net/qqxx6661

拥有专栏:Leetcode题解(Java/Python)、Python爬虫开发

知乎

https://www.zhihu.com/people/yang-zhen-dong-1/

拥有专栏:码农面试助攻手册

掘金

https://juejin.im/user/5b48015ce51d45191462ba55

简书

https://www.jianshu.com/u/b5f225ca2376

我的公众号:Rude3Knife

Java容器(List、Set、Map)知识点快速复习手册(中)

相关文章
相关标签/搜索