百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

web技术分享|LRU 缓存淘汰算法(lru缓存机制)

cac55 2024-10-11 10:51 34 浏览 0 评论

了解 LRU 之前,我们应该了解一下缓存,大家都知道计算机具有缓存内存,可以临时存储最常用的数据,当缓存数据超过一定大小时,系统会进行回收,以便释放出空间来缓存新的数据,但从系统中检索数据的成本比较高。

缓存要求:

  • 固定大小:缓存需要有一些限制来限制内存使用。
  • 快速访问:缓存插入和查找操作应该很快,最好是 O(1) 时间。
  • 在达到内存限制的情况下替换条目:缓存应该具有有效的算法来在内存已满时驱逐条目

如果提供一个缓存替换算法来辅助管理,按照设定的内存大小,删除最少使用的数据,在系统回收之前主动释放出空间,会使得整个检索过程变得非常快,因此 LRU 缓存淘汰算法就出现了。

LRU 原理与实现

[LRU (Least Recently Used) 缓存淘汰算法](https://baike.baidu.com/item/LRU)提出最近被频繁访问的数据应具备更高的留存,淘汰那些不常被访问的数据,即最近使用的数据很大概率将会再次被使用,抛弃最长时间未被访问的数据,目的是为了方便以后获取数据变得更快,例如 Vuekeep-live 组件就是 LRU 的一种实现。

实现的中心思想拆分为以下几步:

  • 新的数据插入到链表头部。
  • 每当缓存命中(即缓存数据被访问),则将数据移到链表头部。
  • 当缓存内存已满时(链表数量已满时),将链表尾部的数据淘汰。

Example

这里使用一个例子来说明 LRU 实现的流程,详细请[参考这里](https://zhuanlan.zhihu.com/p/34989978)。

1. 最开始时,内存空间是空的,因此依次进入A、B、C是没有问题的

2. 当加入D时,就出现了问题,内存空间不够了,因此根据LRU算法,内存空间中A待的时间最为久远,选择A,将其淘汰

3. 当再次引用B时,内存空间中的B又处于活跃状态,而C则变成了内存空间中,近段时间最久未使用的

4. 当再次向内存空间加入E时,这时内存空间又不足了,选择在内存空间中待的最久的C将其淘汰出内存,这时的内存空间存放的对象就是E->B->D

基于双向链表和 HashMap 实现 LRU

常见的 LRU 算法是基于双向链表HashMap 实现的。

双向链表:用于管理缓存数据结点的顺序,新增数据和缓存命中(最近被访问)的数据被放置在 Header 结点,尾部的结点根据内存大小进行淘汰。

HashMap:存储所有结点的数据,当 LRU 缓存命中(进行数据访问)时,进行拦截进行数据置换和删除操作。

双向链表

[双向链表](https://baike.baidu.com/item/%E5%8F%8C%E5%90%91%E9%93%BE%E8%A1%A8/2968731?fr=aladdin)是众多链表中的一种,链表都是采用[链式存储结构](https://baike.baidu.com/item/%E9%93%BE%E5%BC%8F%E5%AD%98%E5%82%A8%E7%BB%93%E6%9E%84),链表中的每一个元素,我们称之为数据结点

每个数据结点都包含一个数据域指针域指针域可以确定结点与结点之间的顺序,通过更新数据结点的指针域的指向可以更新链表的顺序

双向链表的每个数据结点包含一个数据域和两个指针域

  • proir 指向上一个数据结点;
  • data 当前数据结点的数据;
  • next 指向下一个数据结点;

指针域确定链表的顺序,那么双向链表拥有双向指针域,数据结点的之间不在是单一指向,而是双向指向。即 proir 指针域指向上一个数据结点,next 指针域指向下一个数据结点。

同理:

- 单向链表只有一个指针域。

- 循环(环状)链表则是拥有双向指针域,且头部结点的指针域指向尾部结点,尾部结点的指针域指向头部结点。

特殊结点:Header 和 Tailer 结点

链表中还有两个特殊的结点,那就算 Header 结点和 Tailer 结点,分别表示头部结点尾部结点头部结点表示最新的数据或者缓存命中(最近访问过的数据),尾部结点表示长时间未被使用,即将被淘汰的数据节点。

作为算法大家都会关注其时间和空间复杂度 O(n),基于双向链表双向指针域的优势,为了降级时间复杂度,因此为了保证 LRU 新数据和缓存命中的数据都位于链表最前面(Header),缓存淘汰的时候删除最后的结点(Tailer),又要避免数据查找时从头到尾遍历,降低算法的时间复杂度,同时基于双向链表带来的优势,可以改变个别数据结点的指针域从而达到链表数据的更新,如果提供 Header 和 Tailer 结点作为标识的话,可以使用头插法快速增加结点,根据 Tailer 结点也可以在缓存淘汰时快速更新链表的顺序,避免遍历从头到尾遍历,降低算法的时间复杂度。

排序示例

LRU 链表中有 [6,5,4,3,2,1] 6个数据结点,其中 `6` 所在的数据结点为 Header(头部)结点,`1` 所在的数据结点为 Tailer(尾部)结点。如果此时数据 `3` 被访问(缓存命中),`3` 应该被更新至链表头,用数组的思维应该是删除 `3`,但是如果我们利用双向链表双向指针的优势,可以快速的实现链表顺便的更新:

  • `3` 被删除时,`4` 和 `2` 中间没有其他结点,即 `4` 的 `next` 指针域指向 `2` 所在的数据结点;同理,`2` 的 `proir` 指针域指向 `2` 所在的数据结点。

HashMap

至于为什么使用 HashMap,用一句话来概括主要是因为 HashMap 通过 Key 获取速度会快的多,降低算法的时间复杂度。

例如:


  • 我们在 get 缓存的时候从 HashMap 中获取的时候基本上时间复杂度控制在 O(1),如果从链表中一次遍历的话时间复杂度是 O(n)。
  • 我们访问一个已经存在的节点时候,需要将这个节点移动到 header 节点后,这个时候需要在链表中删除这个节点,并重新在 header 后面新增一个节点。这个时候先去 HashMap 中获取这个节点删除节点关系,避免了从链表中遍历,将时间复杂度从 O(N) 减少为 O(1)

由于前端没有 HashMap 的相关 API,我们可以使用 `Object` 或者 `Map` 来代替。

代码实现

现在让我们运用所掌握的数据结构,设计和实现一个,或者参考 [LeeCode 146 题](https://leetcode-cn.com/problems/lru-cache/)。

链表结点 Entry

```typescript

export class Entry<T> {

value: T

key: string | number

next: Entry<T>

prev: Entry<T>

constructor(val: T) {

this.value = val;

}

}

```

双向链表 Double Linked List

主要职责:

  • 管理头部结点和尾部结点
  • 插入新数据时,将新数据移到头部结点
  • 删除数据时,更新删除结点[前后两个结点的指向域](#排序示例)

```typescript

/**

* Simple double linked list. Compared with array, it has O(1) remove operation.

* @constructor

*/

export class LinkedList<T> {

head: Entry<T>

tail: Entry<T>

private _len = 0

/**

* Insert a new value at the tail

*/

insert(val: T): Entry<T> {

const entry = new Entry(val);

this.insertEntry(entry);

return entry;

}

/**

* Insert an entry at the tail

*/

insertEntry(entry: Entry<T>) {

if (!this.head) {

this.head = this.tail = entry;

}

else {

this.tail.next = entry;

entry.prev = this.tail;

entry.next = null;

this.tail = entry;

}

this._len++;

}

/**

* Remove entry.

*/

remove(entry: Entry<T>) {

const prev = entry.prev;

const next = entry.next;

if (prev) {

prev.next = next;

}

else {

// Is head

this.head = next;

}

if (next) {

next.prev = prev;

}

else {

// Is tail

this.tail = prev;

}

entry.next = entry.prev = null;

this._len--;

}

/**

* Get length

*/

len(): number {

return this._len;

}

/**

* Clear list

*/

clear() {

this.head = this.tail = null;

this._len = 0;

}

}

```

LRU 核心算法

主要职责:

  • 将数据添加到链表并更新链表顺序
  • 缓存命中时更新链表的顺序
  • 内存溢出抛弃过时的链表数据

```typescript

/**

* LRU Cache

*/

export default class LRU<T> {

private _list = new LinkedList<T>()

private _maxSize = 10

private _lastRemovedEntry: Entry<T>

private _map: Dictionary<Entry<T>> = {}

constructor(maxSize: number) {

this._maxSize = maxSize;

}

/**

* @return Removed value

*/

put(key: string | number, value: T): T {

const list = this._list;

const map = this._map;

let removed = null;

if (map[key] == null) {

const len = list.len();

// Reuse last removed entry

let entry = this._lastRemovedEntry;

if (len >= this._maxSize && len > 0) {

// Remove the least recently used

const leastUsedEntry = list.head;

list.remove(leastUsedEntry);

delete map[leastUsedEntry.key];

removed = leastUsedEntry.value;

this._lastRemovedEntry = leastUsedEntry;

}

if (entry) {

entry.value = value;

}

else {

entry = new Entry(value);

}

entry.key = key;

list.insertEntry(entry);

map[key] = entry;

}

return removed;

}

get(key: string | number): T {

const entry = this._map[key];

const list = this._list;

if (entry != null) {

// Put the latest used entry in the tail

if (entry !== list.tail) {

list.remove(entry);

list.insertEntry(entry);

}

return entry.value;

}

}

/**

* Clear the cache

*/

clear() {

this._list.clear();

this._map = {};

}

len() {

return this._list.len();

}

}

```

其他 LRU 算法

除了以上常见的 LRU 算法,随着需求的复杂多样,基于 LRU 的思想也衍生出了许多优化算法,例如:

  • LRU-K 算法
  • LRU-Two queues(2Q)算法
  • LRU-Multi queues(MQ)算法
  • [LFU 算法](https://leetcode-cn.com/problems/lfu-cache/)
  • [LRU变种算法](https://blog.csdn.net/u010223431/article/details/105498387)

参考链接

  • [Zrender - LRU](https://github.com/ecomfe/zrender/blob/master/src/core/LRU.ts)
  • [知乎 - 存淘汰算法--LRU算法](https://zhuanlan.zhihu.com/p/34989978)
  • [LRU算法](https://www.cnblogs.com/wyq178/p/9976815.html)
  • [LRU 策略详解和实现](https://leetcode-cn.com/problems/lru-cache/solution/lru-ce-lue-xiang-jie-he-shi-xian-by-labuladong/)

相关推荐

高中生又来卷我们了!手搓 Android 浏览器,可高度定制+脚本支持

回想一下,你曾经的暑假,是怎么度过的?可能是无尽的娱乐时光,或者是懒洋洋的休息日。然而,对于这位Gitee上的高中生来说,他选择在这个暑假里独立开发一款Android浏览器——Vie浏览器,...

网页加载CAD图纸的两个方案对比说明(网页浏览编辑DWG)

一.说明梦想控件提供两种技术在网页中加载CAD图纸,一个是OCX技术方案,另一个是HTML5技术方案,它们各有优缺点,用户需根据实际情况进行选择,下边分别说明一下。1、ocx技术方案(1)OCX技术是...

前后端分离的开源在线考试系统调试实战

开篇在我们的教育生涯中,或多或少的都接触过在线考试系统。例如大学里最常见的各种软件考试,上机考试等,那么有没有开源的这样的系统呢?当然是有了,今天就来调试个开源的在线考试系统。本文重点是调试,因为很多...

网友:小松鼠长大了!UC浏览器推出18周年专版logo引热议

近日,互联网厂商logo更新再次引发热议。作为国内手机浏览器的代表性厂商,UC浏览器的标志性logo小松鼠悄然发生了变化,在网友中引发了关注和讨论。依照UC微博官方账号的说法,这个全新的形象是UC18...

超多案例!谷歌AI模型Nano Banana的5个实用+趣味玩法

再不用这个AI修图神器,你的同行明天就把订单抢光了。谷歌刚放出的NanoBanana,能在一张照片里把背景、姿势、衣服一次换完,脸还是那张脸。实测把地铁照改成海边大片,只用一句话,三秒出图,不用PS来...

2025年最佳Windows数据恢复软件解决方案前5名

您是否正在寻找互联网上排名前五的WindowsPC最佳数据恢复软件解决方案?其实,网上有很多工具可以恢复已删除的文件。但并非所有应用程序都值得使用。值得信赖的文件恢复工具可以帮助您快速检索丢失、删...

电脑数据恢复软件推荐:10个顶级数据恢复软件分享

在数字化的工作与生活中,电脑文件误删除的情况时有发生,这不仅会引发我们的焦虑情绪,更可能导致重要数据的丢失。不过,幸运的是,借助正确的数据恢复软件,我们仍有机会找回那些被误删的文件。10个顶级数据恢复...

更懂国内APP的开源智能体!感知定位推理中文能力全面提升

更懂国内APP的开源智能体!感知定位推理中文能力全面提升“帮我点外卖,别点到广告位。”一句话,说出了多少人对手机自动化的真实期待。浙大和美团刚扔出来的开源项目UItron,就是冲着这句吐槽来的——它真...

美光首家推出采用EUV技术的1γ DDR5 DRAM芯片

美光科技宣布已开始向部分生态系统合作伙伴和客户出货1γ(1-gamma)16GbitDDR5DRAM芯片。美光声称,它是第一个采用1-gamma(1γ)节点的公司,该节点指的是DRAM工艺技术的第...

DDR4的PCB设计及仿真_ddr pcb

以下文章来源于鼎阳硬件智库,作者王彦武DDR4关键技术和方法分析1.1DDR4与DDR3不同之处相对于DDR3,DDR4首先在外表上就有一些变化,比如DDR4将内存下部设计为中间稍微突出,边缘变...

DDR4和DDR5内存的性能差距有哪些?

DDR4和DDR5内存的性能差距主要体现在带宽、延迟、能效及未来扩展性上,以下是关键差异的总结及选择建议:1.带宽与频率DDR4:主流频率为2133MHz–3600MHz,带宽约25.6–30.2...

DDR5内存一根和两根的区别,建议收藏观看。

大家好,我是海韵,DDR5内存条,单条和双条有什么区别,如何选择,DDR5单条和双条内存在性能上存在差距,单条内存保持在64个通道,但内部升级为32乘以2,虽然出口速度相同,但内部运行略有提升,...

Kingston FURY叛逆者DDR5 RGB CUDIMM内存评测 强势突破9000MT/s!

【ZOL中关村在线原创评测】当8000MT/s从当年的液氮超频艰难达成,到如今XMP轻松开启,DDR5内存频率的极限探索似乎看不到终点。在早先,我们曾为大家带来KingstonFURY品牌的叛逆者D...

SK海力士将在年内推出1bnm 32Gb DDR5内存颗粒

IT之家4月25日消息,据韩媒NEWSIS报道,SK海力士在今日的2024年一季度财报电话会议上表示将在年内推出1bnm32GbDDR5内存颗粒。32Gb颗粒意味着消费级的...

DRAM史上最大代际倒挂继续:三星将延长DDR4生产期限至2026年

IT之家8月6日消息,韩媒TheElec今天(8月6日)发布博文,报道称三星决定延长DDR41zDRAM的生产期限至2026年,一方面在DRAM史上最大代际倒挂中进...

取消回复欢迎 发表评论: