0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

如何轻松理解「链表」实现「LRU缓存淘汰算法

电子工程师 来源:lq 2018-12-25 10:09 次阅读

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

三种最常见的链表结构

循环链表的概念

如上图所示:单链表的尾结点指针指向空地址,表示这就是最后的结点了。而循环链表的尾结点指针是指向链表的头结点。

因此循环链表是一种特殊的单链表。它跟单链表唯一的区别就在于尾结点。它像一个环一样首尾相连,所以叫作「循环链表」。

循环链表的特点

和单链表相比,循环链表的优点是从链尾到链头比较方便,当要处理的数据具有环型结构特点时,适合采用循环链表。

双向链表概念

双向链表也叫双链表,是链表的一种,它的链接方向是双向的,它的每个数据结点中都包含有两个指针,分别指向直接后继和直接前驱。

所以,从双向链表中的任意一个结点开始,都可以很方便地访问它的前驱结点和后继结点。

双向链表的数据结构中,会有两个比较重要的参数:pre 和 next。

pre 指向前一个数据结构

next 指向下一个数据结构

单链表与双链表的对比

双向链表的特点

与单链表对比,双链表需要多一个指针用于指向前驱节点,因此如果存储同样多的数据,双向链表要比单链表占用更多的内存空间

双链表的插入和删除需要同时维护 next 和 prev 两个指针。

双链表中的元素访问需要通过顺序访问,支持双向遍历,这就是双向链表操作的灵活性根本

双向链表的基本操作

1.添加元素。

与单向链表相对比双向链表可以在 O(1) 时间复杂度搞定,而单向链表需要 O(n) 的时间复杂度。

双向链表的添加元素包括头插法和尾插法。

头插法和尾插法

头插法:将链表的左边称为链表头部,右边称为链表尾部。头插法是将右边固定,每次新增的元素都在左边头部增加。

尾插法:将链表的左边称为链表头部,右边称为链表尾部。尾插法是将左边固定,每次新增都在链表的右边最尾部。

2.查询元素

查询元素

双向链表的灵活处就是知道链表中的一个元素结构就可以向左或者向右开始遍历查找需要的元素结构。因此对于一个有序链表,双向链表的按值查询的效率比单链表高一些。因为,我们可以记录上次查找的位置 p,每次查询时,根据要查找的值与 p 的大小关系,决定是往前还是往后查找,所以平均只需要查找一半的数据。

3.删除元素

在实际的软件开发中,从链表中删除一个数据无外乎这两种情况:

删除结点中“值等于某个给定值”的结点

删除给定指针指向的结点

删除元素

对于双向链表来说,双向链表中的结点已经保存了前驱结点的指针,删除时不需要像单链表那样遍历。所以,针对第二种情况,单链表删除操作需要 O(n) 的时间复杂度,而双向链表只需要在 O(1) 的时间复杂度。

双向循环链表

双向循环链表

如图所示,双向循环链表的概念很好理解:「双向链表」 + 「循环链表」的组合。

缓存淘汰策略

缓存是一种提高数据读取性能的技术,在硬件设计、软件开发中都有着非常广泛的应用,比如常见的 CPU 缓存、数据库缓存、浏览器缓存等等。

缓存的大小有限,当缓存被用满时,哪些数据应该被清理出去,哪些数据应该被保留?这就需要缓存淘汰策略来决定。常见的策略有三种:先进先出策略 FIFO(First In,First Out)、最少使用策略 LFU(Least Frequently Used)、最近最少使用策略 LRU(Least Recently Used)。

在各个语言的第三方框架中都大量使用到了 LRU 缓存策略。程序员小吴接触到的有Java中的 「 Mybatis 」,iOS中的 「YYCache」与「Lottie」等。

LRU缓存淘汰算法

LRU是最近最少使用策略的缩写,是根据数据的历史访问记录来进行淘汰数据,其核心思想是“如果数据最近被访问过,那么将来被访问的几率也更高”。

LRU概念

链表实现LRU

将Cache的所有位置都用双链表连接起来,当一个位置被命中之后,通过调整链表的指向,将该位置调整到链表头的位置,新加入的Cache直接加到链表头中。

这样,在多次进行Cache操作后,最近被命中的,就会被向链表头方向移动,而没有命中的,而想链表后面移动,链表尾则表示最近最少使用的Cache。

当需要替换内容时候,链表的最后位置就是最少被命中的位置,我们只需要淘汰链表最后的部分即可。

链表实现LRU动画演示

如果此数据之前已经被缓存在链表中了,通过遍历得到这个数据对应的结点,并将其从原来的位置删除,然后再插入到链表的头部。

如果此数据没有在缓存链表中,可以分为两种情况:

如果此时缓存未满,则将此结点直接插入到链表的头部;

如果此时缓存已满,则链表尾结点删除,将新的数据结点插入链表的头部。

链表实现LRU

通过动图可以发现,如果缓存空间足够大,那么存储的数据也就足够多,通过缓存中命中数据的概率就越大,也就提高了代码的执行速度。这就是空间换时间的设计思想。

对于程序开发来说,时间复杂度和空间复杂度是可以相互转化的。说通俗一点,就是:

对于执行的慢的程序,可以通过消耗内存(即构造新的数据结构)来进行优化;

而消耗内存的程序,可以通过消耗时间来降低内存的消耗。

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 数据结构
    +关注

    关注

    3

    文章

    569

    浏览量

    40070
  • 链表
    +关注

    关注

    0

    文章

    80

    浏览量

    10536

原文标题:看动画轻松理解「链表」实现「LRU缓存淘汰算法」

文章出处:【微信号:rgznai100,微信公众号:rgznai100】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    LRU缓存模块最佳实践

    LRU(Least Recently Used)是一种缓存替换算法,它的核心思想是当缓存满时,替换最近最少使用的数据。在实际应用中,LRU
    的头像 发表于 09-30 16:47 841次阅读

    Redis的LRU实现和应用

    在编程中,计数器是一种基本但强大的工具,用于跟踪和管理数据和资源。本文将深入探讨不同类型的计数器的应用,从Redis的LRU(最近最少使用)缓存淘汰算法
    的头像 发表于 12-15 09:24 540次阅读

    【原创】Android开发—Lru核心数据结构实现突破缓存框架

    【原创】Android开发—Lru核心数据结构实现突破缓存框架回复即可获取下载链接[hide=d15]链接:http://pan.baidu.com/s/1c2BfjsW 密码:bta5 更多学习资料加Q:1352716312,
    发表于 06-21 16:58

    《CDN 之我见》系列二:原理篇(缓存、安全)

    Hash 运算都得到同一个余数),则性能与单链表无异,查找时间复杂度是 O(n)。如果磁盘空间不够了怎么办?使用基于访问热度的内容淘汰算法,例如 FIFO、LRU、LFU、SLRU、
    发表于 06-12 16:59

    架构设计应用级缓存回收策略

    下。FIFO(First In First Out):先进先出算法,即先放入缓存的先被移除。LRU(Least Recently Used):最近最少算法,使用时间距离现在最久的那个被
    发表于 01-14 17:08

    算法与数据结构——双向链表

    第三章为算法与数据结构,本文为3.3 双向链表
    的头像 发表于 09-19 17:56 7253次阅读
    <b class='flag-5'>算法</b>与数据结构——双向<b class='flag-5'>链表</b>

    如何进行单链表的查找、插入与删除的详细介绍包括了算法和源程序

    链表的查找、插入与删除。设计算法实现线性结构上的单链表的产生以及元素的查找、插入与删除。具体实现要求:
    发表于 07-16 08:00 22次下载
    如何进行单<b class='flag-5'>链表</b>的查找、插入与删除的详细介绍包括了<b class='flag-5'>算法</b>和源程序

    带你轻松理解数据结构与算法系列

      主要使用图片来描述常见的数据结构和算法轻松阅读并理解掌握。本系列包括各种堆、各种队列、各种列表、各种树、各种图、各种排序等等。
    发表于 08-01 17:34 2次下载
    带你<b class='flag-5'>轻松</b><b class='flag-5'>理解</b>数据结构与<b class='flag-5'>算法</b>系列

    一文读懂缓存淘汰算法:LFU 算法

    实现难度上来说,LFU 算法的难度大于 LRU 算法,因为 LRU 算法相当于把数据按照时间排
    的头像 发表于 08-25 17:37 9838次阅读
    一文读懂<b class='flag-5'>缓存</b><b class='flag-5'>淘汰</b><b class='flag-5'>算法</b>:LFU <b class='flag-5'>算法</b>

    谷歌在内存方面依赖于per memcg lru lock

    能力。 作为世间最流行的操作系统Linux, 内核使用LRU, Last Recent Used 链表来管理全部用户使用的内存,用一组链表串联起一个个的内存页,并且使用lru lock
    的头像 发表于 01-15 14:00 1866次阅读
    谷歌在内存方面依赖于per memcg <b class='flag-5'>lru</b> lock

    设计并实现一个满足LRU约束的数据结构

    LRUCache(int capacity)` 以 **「正整数」** 作为容量 `capacity` 初始化 `LRU` 缓存
    的头像 发表于 06-07 17:05 946次阅读
    设计并<b class='flag-5'>实现</b>一个满足<b class='flag-5'>LRU</b>约束的数据结构

    基于LRU-K模型如何实现高效的元数据缓存

    对于存储来说,性能是绕不开的话题。当提到性能,可靠、高效的缓存策略是极其重要的。
    的头像 发表于 06-29 15:05 832次阅读
    基于<b class='flag-5'>LRU</b>-K模型如何<b class='flag-5'>实现</b>高效的元数据<b class='flag-5'>缓存</b>?

    Redis的LRU与LFU算法实现

    Redis是一款基于内存的高性能NoSQL数据库,数据都缓存在内存里, 这使得Redis可以每秒轻松地处理数万的读写请求。
    的头像 发表于 07-11 09:48 1016次阅读
    Redis的<b class='flag-5'>LRU</b>与LFU<b class='flag-5'>算法</b><b class='flag-5'>实现</b>

    链表和双链表的区别在哪里

    。 上面的三幅图对于理解链表的插入、删除很重要,看代码的时候要对着看。 实际中经常使用的一般为带头双向循环链表,下面是一个双向循环链表的 demo,是最简单的情况。
    的头像 发表于 07-27 11:20 1585次阅读
    单<b class='flag-5'>链表</b>和双<b class='flag-5'>链表</b>的区别在哪里

    redis的lru原理

    Redis是一种基于内存的键值数据库,它使用了LRU(Least Recently Used)算法来进行缓存的数据淘汰LRU
    的头像 发表于 12-05 09:56 587次阅读