在最近的面试中,我曾被多次问到,怎么实现一个最近最少使用(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;

  }

}


更多相关文章

  1. Mybaitis缓存的优化
  2. MyBatis 延迟加载、一二级缓存、架构设计的面试题(常问,重点了解)
  3. 帮你解读什么是Redis缓存穿透和缓存雪崩
  4. MyBatis之Mapper XML 文件详解(六)-缓存配置
  5. redis缓存服务
  6. HTTP缓存
  7. 10 行 Java 代码实现最近被使用( LRU )缓存
  8. 用 cURL 请求测试 ETag 浏览器缓存[每日前端夜话0xCC]

随机推荐

  1. 解决android调用系统相机拍照保存时onAct
  2. 路径提供者文档目录是一个安全的位置吗?
  3. 如何在Android 7.0+中检索SD卡的序列号?
  4. Android多线程下载远程图片【转】
  5. 转:Android Studio Error:Connection time
  6. MD5加密,java工具类 String 转变成MD5 St
  7. 没有包lib32z1,lib32ncurses5,lib32stdc+
  8. Android TextView设置字体风格
  9. 如何在Fragment中的WebView中添加“Go Ba
  10. Android自定义控件——开源组件SlidingMe