作者:_code_x
链接:https://www.jianshu.com/p/b7fed77324b9

写在前

就是一种缓存淘汰策略。

计算机的缓存容量有限,如果缓存满了就要删除一些内容,给新内容腾位置。但问题是,删除哪些内容呢?我们肯定希望删掉哪些没什么用的缓存,而把有用的数据继续留在缓存里,方便之后继续使用。那么,什么样的数据,我们判定为「有用的」的数据呢?

LRU 缓存淘汰算法就是一种常用策略。LRU 的全称是 Least Recently Used,也就是说我们认为最近使用过的数据应该是是「有用的」,很久都没用过的数据应该是无用的,内存满了就优先删那些很久没用过的数据。

算法描述

运用你所掌握的数据结构,设计和实现一个 LRU (最近最少使用) 缓存机制

实现 LRUCache 类:

  • LRUCache(int capacity) 以正整数作为容量 capacity 初始化 LRU 缓存

  • int get(int key) 如果关键字 key 存在于缓存中,则返回关键字的值,否则返回 -1 。

  • void put(int key, int value) 如果关键字已经存在,则变更其数据值;如果关键字不存在,则插入该组「关键字-值」。当缓存容量达到上限时,它应该在写入新数据之前删除最久未使用的数据值,从而为新的数据值留出空间。

注意哦,get 和 put 方法必须都是 O(1) 的时间复杂度!

示例

/* 缓存容量为 2 */
LRUCache cache = new LRUCache(2);
// 你可以把 cache 理解成一个队列
// 假设左边是队头,右边是队尾
// 最近使用的排在队头,久未使用的排在队尾
// 圆括号表示键值对 (key, val)cache.put(1, 1);
// cache = [(1, 1)]
cache.put(2, 2);
// cache = [(2, 2), (1, 1)]
cache.get(1);       // 返回 1
// cache = [(1, 1), (2, 2)]
// 解释:因为最近访问了键 1,所以提前至队头
// 返回键 1 对应的值 1
cache.put(3, 3);
// cache = [(3, 3), (1, 1)]
// 解释:缓存容量已满,需要删除内容空出位置
// 优先删除久未使用的数据,也就是队尾的数据
// 然后把新的数据插入队头
cache.get(2);       // 返回 -1 (未找到)
// cache = [(3, 3), (1, 1)]
// 解释:cache 中不存在键为 2 的数据
cache.put(1, 4);
// cache = [(1, 4), (3, 3)]
// 解释:键 1 已存在,把原始值 1 覆盖为 4
// 不要忘了也要将键值对提前到队头

算法设计

分析上面的操作过程,要让 put 和 get 方法的时间复杂度为 O(1),我们可以总结出 cache 这个数据结构必要的条件:查找快,插入快,删除快,有顺序之分。

因为显然 cache 必须有顺序之分,以区分最近使用的和久未使用的数据;而且我们要在 cache 中查找键是否已存在;如果容量满了要删除最后一个数据;每次访问还要把数据插入到队头。

那么,什么数据结构同时符合上述条件呢?哈希表查找快,但是数据无固定顺序;链表有顺序之分,插入删除快,但是查找慢。所以结合一下,形成一种新的数据结构:哈希链表。

双向链表也叫双链表,是链表的一种,它的每个数据结点中都有两个指针,分别指向直接后继和直接前驱。所以,从双向链表中的任意一个结点开始,都可以很方便地访问它的前驱结点和后继结点。一般我们都构造双向循环链表。

LRU 缓存算法的核心数据结构就是哈希链表:双向链表和哈希表的结合体。这个数据结构长这样:

思想很简单,就是借助哈希表赋予了链表快速查找的特性嘛:可以快速查找某个 key 是否存在缓存(链表)中,同时可以快速删除、添加节点。回想刚才的例子,这种数据结构是不是完美解决了 LRU 缓存的需求?

代码实现

  • 首先定义双端链表类(包括数据和记录前驱/后继节点的指针

class DLinkedNode {int key;int value;DLinkedNode pre;DLinkedNode next;public DLinkedNode() {};public DLinkedNode(int key, int value) {this.key = key;this.value = value;}
}
  • 双向链表需要提供一些接口api,便于我们操作,主要就是链表的一些操作,画图理解!

private void addFirst(DLinkedNode node) {node.pre = head;node.next = head.next;head.next.pre = node;head.next = node;
}private void moveToFirst(DLinkedNode node) {remove(node);addFirst(node);
}private void remove(DLinkedNode node) {node.pre.next = node.next;node.next.pre = node.pre;
}// 删除尾结点,并返回头节点
private DLinkedNode removeLast() {DLinkedNode ans = tail.pre;remove(ans);return ans;
}private int getSize() {return size;
}
  • 确定LRU缓存类的成员变量(链表长度、缓存容量和map映射等)和构造函数。注意:定义虚拟头尾结点便于在头部插入元素或者寻找尾部元素!并在构造函数初始化。

private Map<Integer, DLinkedNode> cache = new HashMap<>();
private int size;
private int capacity;
private DLinkedNode head, tail;public LRUCache(int capacity) {this.size = 0;this.capacity = capacity;head = new DLinkedNode();tail = new DLinkedNode();head.next = tail;tail.pre = head;
}
  • 核心代码:get和put方法,都是先根据key获取这个映射,根据映射节点的情况(有无)进行操作。注意:

    • get和put都在使用,所以数据要提前!

    • put操作如果改变了双端链表长度(不是仅改变值),需要先判断是否达到最大容量!

public int get(int key) {DLinkedNode node = cache.get(key);if (node == null) {return -1;}// 将该数据移到双端队列头部moveToFirst(node);return node.value;
}public void put(int key, int value) {DLinkedNode node = cache.get(key);if (node != null) {// 如果存在key,先修改值,然后移动到头部node.value = value;moveToFirst(node);} else {// 如果key存在,先考虑是否超过容量限制if (capacity == cache.size()) {// 删除尾结点和hash表中对应的映射!DLinkedNode tail = removeLast();cache.remove(tail.key);--size;}DLinkedNode newNode = new DLinkedNode(key, value);// 建立映射,并更新双向链表头部cache.put(key, newNode);addFirst(newNode);++size;}
}

完整代码如下:

class LRUCache {class DLinkedNode {int key;int value;DLinkedNode pre;DLinkedNode next;public DLinkedNode() {};public DLinkedNode(int key, int value) {this.key = key;this.value = value;}}private Map<Integer, DLinkedNode> cache = new HashMap<>();private int size;private int capacity;// 虚拟头尾结点便于在头部插入元素或者寻找尾部元素!private DLinkedNode head, tail;public LRUCache(int capacity) {this.size = 0;this.capacity = capacity;// 使用伪头部和伪尾部节点head = new DLinkedNode();tail = new DLinkedNode();head.next = tail;tail.pre = head;}public int get(int key) {DLinkedNode node = cache.get(key);if (node == null) {return -1;}// 将该数据移到双端队列头部moveToFirst(node);return node.value;}public void put(int key, int value) {DLinkedNode node = cache.get(key);if (node != null) {// 如果存在key,先修改值,然后移动到头部node.value = value;moveToFirst(node);} else {// 如果key存在,先考虑是否超过容量限制if (capacity == cache.size()) {// 删除尾结点和hash表中对应的映射!DLinkedNode tail = removeLast();cache.remove(tail.key);--size;}DLinkedNode newNode = new DLinkedNode(key, value);// 建立映射,并更新双向链表头部cache.put(key, newNode);addFirst(newNode);++size;}}private void addFirst(DLinkedNode node) {node.pre = head;node.next = head.next;head.next.pre = node;head.next = node;}private void moveToFirst(DLinkedNode node) {remove(node);addFirst(node);}private void remove(DLinkedNode node) {node.pre.next = node.next;node.next.pre = node.pre;}// 删除尾结点,并返回头节点private DLinkedNode removeLast() {DLinkedNode ans = tail.pre;remove(ans);return ans;}private int getSize() {return size;}
}

总结与补充

  • LRU缓存机制的核心:双向链表(保证元素有序,且能快速的插入和删除)+hash表(可以快速查询)

  • 为什么使用双向链表?因为:对于删除操作,使用双向链表,我们可以在O(1)的时间复杂度下,找到被删除节点的前节点。

  • 为什么要在链表中同时存键值,而不是只存值?因为:当缓存容量满了之后,我们不仅要在双向链表中删除最后一个节点(即最久没有使用的节点),还要把cache中映射到该节点的key删除,这个key只能有Node得到(即hash表不能通过值得到键)。

巨人的肩膀

https://leetcode-cn.com/problems/lru-cache/solution/lru-ce-lue-xiang-jie-he-shi-xian-by-labuladong/

算法必知 --- LRU缓存淘汰算法相关推荐

  1. 看动画轻松理解「链表」实现「LRU缓存淘汰算法」

    作者 | 程序员小吴,哈工大学渣,目前正在学算法,开源项目 「 LeetCodeAnimation 」5500star,GitHub Trending 榜连续一月第一. 本文为 AI科技大本营投稿文章 ...

  2. 数据结构与算法 / LRU 缓存淘汰算法

    一.诞生原因 缓存是一种提供数据读取性能的技术,在硬件设计.软件开发中有广泛的应用,比如常见的 CPU 缓存,DB 缓存和浏览器缓存等.但是缓存的大小是有限的,需要一定的机制判断哪些数据需要淘汰,即: ...

  3. 06 | 链表(上):如何实现LRU缓存淘汰算法?

    缓存 作用 缓存是一种提高数据读取性能的技术,在硬件设计.软件开发中都有着非常广泛的应用,比如常见的 CPU 缓存.数据库缓存.浏览器缓存等等. 淘汰策略 常见的策略有三种:先进先出策略 FIFO(F ...

  4. leetcode刷题:LRU缓存淘汰算法

    题目: 分析: 计算机的缓存容量有限,如果缓存满了就要删除一些内容,给新内容腾位置.但问题是,删除哪些内容呢?我们肯定希望删掉哪些没什么用的缓存,而把有用的数据继续留在缓存里,方便之后继续使用.那么, ...

  5. 看动画轻松理解「链表」实现「 LRU 缓存淘汰算法」

    作者 | 吴至波 责编 | 胡巍巍 快速挑战Python全栈工程师: https://edu.csdn.net/topic/python115?utm_source=csdn_bw 前几节学习了「链表 ...

  6. 看动画理解「链表」实现LRU缓存淘汰算法

    前几节学习了「链表」.「时间与空间复杂度」的概念,本节将结合「循环链表」.「双向链表」与 「用空间换时间的设计思想」来设计一个很有意思的缓存淘汰策略:LRU缓存淘汰算法. 循环链表的概念 如上图所示: ...

  7. LRU缓存淘汰算法优化

    上文中提到了LRU 缓存淘汰算法,可以帮助我们更好更合理的去使用缓存.但是它也有一个缺点就是如果有一些不满足"如果数据最近被访问过,那么将来被访问的几率也更高"的规律时,会破坏缓存 ...

  8. 如何基于链表实现 LRU 缓存淘汰算法?

    什么是LRU LRU(Least recently used,最近最少使用)算法根据数据的历史访问记录来进行淘汰数据,其核心思想是"如果数据最近被访问过,那么将来被访问的几率也更高" ...

  9. Python实现:详解LRU缓存淘汰算法

    大家好,今天我们和大家聊一个非常常用的算法,叫做LRU. LRU的英文全称是Least Recently Used,也即最不经常使用.我们看着好像挺迷糊的,其实这个含义要结合缓存一起使用.对于工程而言 ...

最新文章

  1. Ubuntu服务器版硬件认证详情
  2. 线程同步-AutoResetEvent
  3. 这个在Github有52100颗星星的项目,怎么还有人不知道鸭!
  4. 图像超分辨率增强ESRGAN运行教程,有数据
  5. jvisualvm/Jconsole监控WAS(WebSphere)中间件
  6. 2021 年百度之星·程序设计大赛 - 初赛二
  7. 19 接口类和抽象类
  8. 定积分证明题例题_数列极限求法十五种(25个例题+推文送给微积分和数学分析同学)...
  9. 传教士与野人问题深度优先搜索算法(DFS)-Python实现
  10. 使用vue-admin-template基础模板开发后台管理系统必会技能
  11. 【目标检测】SSD(Single Shot MultiBox Detector)的复现
  12. 命令行窗口对MySQL的基本使用
  13. 大学“电路分析基础”试题合集第四章
  14. 如何修改C盘用户目录下的用户名
  15. redis安装、持久化、数据类型、常用操作、操作键值、安全设置、慢查询日志、存储session、主从配置、集群介绍、集群搭建配置、集群操作,php安装redis扩展...
  16. Linux文件系统:minix文件系统二进制分析3(硬链接与软链接)
  17. 出现RPC服务器不可用的解决方法
  18. 基于XMPP的即时通信系统的建立(二)— XMPP详解
  19. 数学建模-相关性分析(Matlab)
  20. Navicat的1577 – Cannot proceed because system tables used by Event Scheduler where found damaged at s

热门文章

  1. php worker微信,微信小程序API createWorker
  2. 如何扫描和修复 Linux 磁盘错误
  3. DAIC:针对BEC欺诈的开源利器
  4. 7/15 我的第一篇博客-写给自己
  5. poj3694(双联通分量+LCA)
  6. HDU1808(抽屉原理)
  7. 数位dp ---- 暴力 + 二进制的数位dp 2020济南 L Bit Sequence
  8. jupyter 写入csv pandas_Pandas 最详细教程在这里
  9. 解题报告 - 牛客练习赛63 C - 牛牛的揠苗助长(货仓选址+二分)
  10. UVA1103 古代象形符号 Ancient Messages解题报告(DFS,字符串)难度⭐⭐⭐⭐