一:概述

  - LRU 用于管理缓存策略,其本身在 Linux/Redis/Mysql 中均有实现。只是实现方式不尽相同。

  - LRU 算法【Least recently used(最近最少使用)】

  - 根据数据的历史访问记录来进行淘汰数据,其核心思想是 "如果数据最近被访问过,那么将来被访问的几率也更高"。

   - GITHUB

二:单链表实现的 Lru 算法

  - 思路

    - 单链表实现

  - 流程

    - 1. 新数据插入到链表头部;

    - 2. 每当缓存命中(即缓存数据被访问),则将数据移到链表头部;

    - 3. 当链表满的时候,将链表尾部的数据丢弃。

  - 优点

    - 实现简单。

    - 当存在热点数据时,LRU的效率很好。

  - 缺点

    - 偶发性的、周期性的批量操作会导致LRU命中率急剧下降,缓存污染情况比较严重。

    - 命中时需要遍历链表,找到命中的数据块索引,然后需要将数据移到头部。

  - 流程图

    - 

  - 代码实现(PHP)

    • /**
      * 思路
      * 单链表实现
      * 原理
      * 单链表
      * 流程
      * 1. 新数据插入到链表头部;
      * 2. 每当缓存命中(即缓存数据被访问),则将数据移到链表头部;
      * 3. 当链表满的时候,将链表尾部的数据丢弃。
      * 优点
      * 实现简单。
      * 当存在热点数据时,LRU的效率很好。
      * 缺点
      * 偶发性的、周期性的批量操作会导致LRU命中率急剧下降,缓存污染情况比较严重。
      * 命中时需要遍历链表,找到命中的数据块索引,然后需要将数据移到头部。
      */
      class Lru
      {
      public static $lruList = []; // 顺序存储单链表结构
      public static $maxLen = ; // 链表允许最大长度
      public static $nowLen = ; // 链表当前长度 /**
      * LRU_1 constructor.
      * 由于 PHP 不是常驻进程程序,所以链表初始化可以通过 Mysql/Redis 实现
      */
      public function __construct()
      {
      self::$lruList = [];
      self::$nowLen = count(self::$lruList);
      } /**
      * 获取 key => value
      * @param $key
      * @return null
      */
      public function get($key)
      {
      $value = null; // lru 队列为空,直接返回
      if (!self::$lruList) {
      self::$lruList[] = [$key => $this->getData($key)]; // 根据实际项目情况获取数据
      self::$nowLen++;
      return $value;
      } // 查找 lru 缓存
      for ($i = ; $i < self::$nowLen; $i++) { // 如果存在缓存,则直接返回,并将数据重新插入链表头部
      if (isset(self::$lruList[$i][$key])) { $value = self::$lruList[$i][$key]; unset(self::$lruList[$i]); array_unshift(self::$lruList, [$key => $value]); break;
      }
      } // 如果没有找到 lru 缓存
      if (!isset($value)) { // 插入头部
      array_unshift(self::$lruList, [$key => $this->getData($key)]); // 根据实际项目情况获取数据
      self::$nowLen++; if (self::$nowLen > self::$maxLen) {
      self::$nowLen--;
      array_pop(self::$lruList);
      }
      } return $value; } /**
      * 输出 Lru 队列
      */
      public function echoLruList()
      {
      var_dump(self::$lruList);
      } /**
      * 根据真实环境获取数据
      * @param $key
      * @return string
      */
      public function getData($key)
      {
      return 'data';
      }
      }

三:Lru K 算法

  - 思路

    - 为了避免 LRU 的 '缓存污染' 问题

    - 增加一个队列来维护缓存出现的次数。其核心思想是将“最近使用过1次”的判断标准扩展为“最近使用过K次”。

  - 原理

    - 相比LRU,LRU-K需要多维护一个队列,用于记录所有缓存数据被访问的历史。

    - 只有当数据的访问次数达到K次的时候,才将数据放入缓存。

    - 当需要淘汰数据时,LRU-K会淘汰第K次访问时间距当前时间最大的数据.

  - 流程

    - 1.数据第一次被访问,加入到访问历史列表;

    - 2.如果数据在访问历史列表里后没有达到K次访问,则按照一定规则 LRU淘汰;

    - 3.当访问历史队列中的数据访问次数达到K次后,将数据索引从历史队列删除,将数据移到缓存队列中,并缓存此数据,缓存队列重新按照时间排序;

    - 4.缓存数据队列中被再次访问后,重新排序;

    - 5.需要淘汰数据时,淘汰缓存队列中排在末尾的数据,即:淘汰“倒数第K次访问离现在最久”的数据。

  - 优点

    - LRU-K降低了“缓存污染”带来的问题,命中率比LRU要高。

  - 缺点

    - LRU-K队列是一个优先级队列,算法复杂度和代价比较高。

    - 由于LRU-K还需要维护历史队列,所以消耗的内存会更多。

  - 流程图

    - 

  - 代码实现

    • /**
      * 思路
      * 为了避免 LRU 的 '缓存污染' 问题
      * 增加一个队列来维护缓存出现的次数。其核心思想是将“最近使用过1次”的判断标准扩展为“最近使用过K次”。
      * 原理
      * 相比LRU,LRU-K需要多维护一个队列,用于记录所有缓存数据被访问的历史。
      * 只有当数据的访问次数达到K次的时候,才将数据放入缓存。
      * 当需要淘汰数据时,LRU-K会淘汰第K次访问时间距当前时间最大的数据
      * 流程
      * 1.数据第一次被访问,加入到访问历史列表;
      * 2.如果数据在访问历史列表里后没有达到K次访问,则按照一定规则 LRU淘汰;
      * 3.当访问历史队列中的数据访问次数达到K次后,将数据索引从历史队列删除,将数据移到缓存队列中,并缓存此数据,缓存队列重新按照时间排序;
      * 4.缓存数据队列中被再次访问后,重新排序;
      * 5.需要淘汰数据时,淘汰缓存队列中排在末尾的数据,即:淘汰“倒数第K次访问离现在最久”的数据。
      * 优点
      * LRU-K降低了“缓存污染”带来的问题,命中率比LRU要高。
      * 缺点
      * LRU-K队列是一个优先级队列,算法复杂度和代价比较高。
      * 由于LRU-K还需要维护历史队列,所以消耗的内存会更多。
      */
      class Lru_K
      {
      public static $historyList = []; // 访问历史队列
      public static $lruList = []; // 顺序存储单链表结构
      public static $maxLen = ; // 链表允许最大长度
      public static $nowLen = ; // 链表当前长度 /**
      * LRU_K constructor.
      * 由于 PHP 不是常驻进程程序,所以链表初始化可以通过 Mysql/Redis 实现
      */
      public function __construct()
      {
      self::$lruList = [];
      self::$historyList = [];
      self::$nowLen = count(self::$lruList);
      } /**
      * 获取 key => value
      * @param $key
      * @return null
      */
      public function get($key)
      {
      $value = null; // 查找 lru 缓存
      for ($i = ; $i < self::$nowLen; $i++) { // 如果存在缓存,则直接返回,并将数据重新插入链表头部
      if (isset(self::$lruList[$i][$key])) { $value = self::$lruList[$i][$key]; unset(self::$lruList[$i]); array_unshift(self::$lruList, [$key => $value]); break;
      }
      } // 如果没有找到 lru 缓存, 则进入历史队列进行计数,当次数大于等于5时候,进入缓存队列
      if (!isset($value)) {
      self::$historyList[$key]++; $value = $this->getData($key); // 进入缓存队列
      if (self::$historyList[$key] >= ) {
      array_unshift(self::$lruList, [$key => $value]); // 根据实际项目情况获取数据
      self::$nowLen++; if (self::$nowLen > self::$maxLen) {
      self::$nowLen--;
      array_pop(self::$lruList);
      } unset(self::$historyList[$key]); // 历史队列推出
      }
      } return $value; } /**
      * 输出 Lru 队列
      */
      public function echoLruList()
      {
      var_dump(self::$lruList);
      var_dump(self::$historyList);
      } /**
      * 根据真实环境获取数据
      * @param $key
      * @return string
      */
      public function getData($key)
      {
      return 'data';
      }
      }

《算法 - Lru算法》的更多相关文章

  1. 简单物联网:外网访问内网路由器下树莓派Flask服务器

    最近做一个小东西,大概过程就是想在教室,宿舍控制实验室的一些设备. 已经在树莓上搭了一个轻量的flask服务器,在实验室的路由器下,任何设备都是可以访问的:但是有一些限制条件,比如我想在宿舍控制我种花 ...

  2. 利用ssh反向代理以及autossh实现从外网连接内网服务器

    前言 最近遇到这样一个问题,我在实验室架设了一台服务器,给师弟或者小伙伴练习Linux用,然后平时在实验室这边直接连接是没有问题的,都是内网嘛.但是回到宿舍问题出来了,使用校园网的童鞋还是能连接上,使 ...

  3. 外网访问内网Docker容器

    外网访问内网Docker容器 本地安装了Docker容器,只能在局域网内访问,怎样从外网也能访问本地Docker容器? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动Docker容器 ...

  4. 外网访问内网SpringBoot

    外网访问内网SpringBoot 本地安装了SpringBoot,只能在局域网内访问,怎样从外网也能访问本地SpringBoot? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装Java 1 ...

  5. 外网访问内网Elasticsearch WEB

    外网访问内网Elasticsearch WEB 本地安装了Elasticsearch,只能在局域网内访问其WEB,怎样从外网也能访问本地Elasticsearch? 本文将介绍具体的实现步骤. 1. ...

  6. 怎样从外网访问内网Rails

    外网访问内网Rails 本地安装了Rails,只能在局域网内访问,怎样从外网也能访问本地Rails? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动Rails 默认安装的Rails端口 ...

  7. 怎样从外网访问内网Memcached数据库

    外网访问内网Memcached数据库 本地安装了Memcached数据库,只能在局域网内访问,怎样从外网也能访问本地Memcached数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装 ...

  8. 怎样从外网访问内网CouchDB数据库

    外网访问内网CouchDB数据库 本地安装了CouchDB数据库,只能在局域网内访问,怎样从外网也能访问本地CouchDB数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动Cou ...

  9. 怎样从外网访问内网DB2数据库

    外网访问内网DB2数据库 本地安装了DB2数据库,只能在局域网内访问,怎样从外网也能访问本地DB2数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动DB2数据库 默认安装的DB2 ...

  10. 怎样从外网访问内网OpenLDAP数据库

    外网访问内网OpenLDAP数据库 本地安装了OpenLDAP数据库,只能在局域网内访问,怎样从外网也能访问本地OpenLDAP数据库? 本文将介绍具体的实现步骤. 1. 准备工作 1.1 安装并启动 ...

随机推荐

  1. OLED液晶屏幕(1)OLED液晶屏幕ssd1306驱动芯片 arduino运行 ESP8266-07可以 12f不可以

    OLED屏幕有各种形状和尺寸,但目前有两种非常受欢迎的屏幕尺寸. 1)0.96“ 2)1.3“ 他们也有2种常见的颜色 1)蓝色 2)白色 驱动OLED的芯片常用的有两种.这两种芯片有许多非常相似的设 ...

  2. Lightning Web Components 组件生命周期(六)

    组件创建以及渲染流程 组件移除dom 处理流程 组件从dom 移除 组件中的disconnectedCallback() 方法被调用 子组件从dom 移除 每个子组件的disconnectedCall ...

  3. HTML相对路径和绝对路径

     一.相对路径 相对路径不带有盘符,通常是以HTML网页文件为起点,通过层级关系描述目标图像的位置. 相对于文件自身出发,就是相对路径. 主要有以下几种情况:   ①文件和图片(html文档)在同一个 ...

  4. 【NOIP2014】真题回顾

    题目链接 生活大爆炸版石头剪刀布 就是个模拟,不说了 联合权值 枚举每个点,统计它任意两个儿子的联合权值,统计的时候维护sum和max就行了 飞扬的小鸟 比较好的DP题,不难想到用dp[i][j]表示 ...

  5. Cocos Creator 功能介绍

    cc.Class({ extends: cc.Component, properties: { anim: cc.Animation }, playRun: function() { this.ani ...

  6. c博客作业-我的第一篇博客

    1.你对网络专业或者计算机专业了解是怎样的? 以前接触计算机,只是把它当作娱乐的工具,并没有太过了解,现在我通过查阅了解了一些计算机的知识. 计算机专业的学生要学习的不仅是会使用,而且要学习计算机的基 ...

  7. err Invalid input of type: 'dict'. Convert to a byte, string or number first

    一个问题引发的血案: 用python向redis写入数据报错: redis.exceptions.DataError: Invalid input of type: 'dict'. Convert t ...

  8. mysql abs() 获取绝对值

    mysql> -); +----------+ | abs(-) | +----------+ | | +----------+ row in set (0.00 sec)

  9. 【POJ3083】Children of the Candy Corn

    本题传送门 本题知识点:深度优先搜索 + 宽度优先搜索 本题题意是求三个路径长度,第一个是一直往左手边走的距离,第二个是一直往右手边走的距离,第三个是最短距离. 第三个很好办,就是一个简单的bfs的模 ...

  10. windows环境搭建dubbo服务

    windows环境搭建dubbo服务 1:首先需要下载dubbo的注册中心 zookeeper zookeeper注册中心下载地址链接:http://mirror.bit.edu.cn/apache/ ...