10 行 Java 代码实现最近被使用(LRU)缓存
(点击
上方公众号
,可快速关注)
来源:ImportNew - paddx
在最近的面试中,我曾被多次问到,怎么实现一个最近最少使用(LRU)的缓存。缓存可以通过哈希表来实现,然而为这个缓存增加大小限制会变成另一个有意思的问题。现在我们看一下怎么实现。
最近最少使用缓存的回收
为了实现缓存回收,我们需要很容易做到:
查询出最近最晚使用的项
给最近使用的项做一个标记
链表可以实现这两个操作。检测最近最少使用的项只需要返回链表的尾部。标记一项为最近使用的项只需要从当前位置移除,然后将该项放置到头部。比较困难的事情是怎么快速的在链表中找到该项。
哈希表的帮助
看一下我们工具箱中的数据结构,哈希表可以在(消耗)常量的时间内索引到某个对象。如果我们创建一个形如key->链表节点的哈希表,我们就能够在常量时间内找到最近使用的节点。更甚的是,我们也能够在常量时间内判断节点的是否存在(或不存在);
找到这个节点后,我们就能将这个节点移动到链表的最前端,标记为最近使用的项了。
Java的捷径
据我所知,很少有一种编程语言的标准库中有通用的数据结构能提供上述功能的。这是一种混合的数据结构,我们需要在哈希表的基础上建立一个链表。但是Java已经为我们提供了这种形式的数据结构-LinkedHashMap!它甚至提供可覆盖回收策略的方法(见
removeEldestEntry文档
)。唯一需要我们注意的事情是,改链表的顺序是插入的顺序,而不是访问的顺序。但是,有一个构造函数提供了一个选项,可以使用访问的顺序(见文档
)。removeEldestEntry文档http://docs.oracle.com/javase/7/docs/api/java/util/LinkedHashMap.html#removeEldestEntry(java.util.Map.Entry)
文档http://docs.oracle.com/javase/7/docs/api/java/util/LinkedHashMap.html#LinkedHashMap(int,%20float,%20boolean)
无需多说:
import java.util.LinkedHashMap;
import java.util.Map;
public LRUCache<K, V> extends LinkedHashMap<K, V> {
private int cacheSize;
public LRUCache(int cacheSize) {
super(16, 0.75, true);
this.cacheSize = cacheSize;
}
protected boolean removeEldestEntry(Map.Entry<K, V> eldest) {
return size() >= cacheSize;
}
}
看完本文有收获?请转发分享给更多人
关注「ImportNew」,提升Java技能
- 司法部:全国司法所基本实现对乡镇、街道覆盖
- 首艘“中国造”极地破冰船“雪龙2”号实现WiFi覆盖
- 革命文化是奋力实现民族复兴的持久动力
- 实现农村污水处理全达标
- 贫困人口基本养老保险将实现全覆盖
- 创业者程颢:选择主动人生 实现自我价值
- 就在明天!一元购车的大惊喜将在贵州这个地方实现,那个幸运儿会
- 澳洲又出抗衰老神药:5年内或可实现延长人类寿命到150岁!而且是
- 为20名贫困儿童实现开学心愿
- 习近平:支持非洲参与共建“一带一路”,共享发展成果,实现共同