面试的时候问到候选人 Redis 相关问题时,发现一个现象:一部分候选人分不清 Redis 的「键过期策略」和「内存淘汰策略」。今天就来说一说这老哥俩。

简单来说,过期策略就是当 key 到了指定的过期时间后,Redis 是用什么方式将其删除的;而淘汰策略指的是当内存不够用时,Redis 如何处理。

过期策略

Redis 的键过期方式有两种:被动和主动。

**被动方式:**当客户端尝试访问某个过期 key 时,Redis 发现该 key 已过期,将其删除。

**主动方式:**为每个设置了过期时间的 key 设置一个定时器,当到达过期时间时将 key 删除。

被动方式的问题很明显,可能有些 key 很久不会被访问,甚至永远都不会被访问,浪费内存资源。

主动方式也有问题,需要不断的去查看 key 是否已经过期,过于消耗 CPU 资源。

还有一种定期删除的方式,比如 30s 对设置过期时间的 key 进行一次扫描,并删除那些已经过期的 key。这种方式看起来是结合前两者的优势,但是它还是过于粗暴。

那么,Redis 到底是如何处理过期 key 的呢?答案是,被动 + 优化版的定期删除。被动已经说过了,下面看看 Redis 是怎么优化定期删除的。

Redis 是如何做的:

  1. 随机选择 20 个设置带有过期时间的 key
  2. 删除其中已经到达过期时间的 key
  3. 如果 20 个被选 key 中有超过 1/4(5 个)已经过期,则重复步骤 1,直到过期 key 低于 1/4

以上步骤的执行频率为每秒 10 次。

Redis 通过这种被动 + 优化版定期删除的方式使得内存和 CPU 资源的占用达到一个平衡状态,既不会让无效 key 占用过多内存,又没有过多的消耗 CPU 去做扫描工作。

过期精度

在 Redis 2.4 及以前版本,过期时间可能不是十分准确,有 1 秒以内的误差。

从 Redis 2.6 起,过期时间误差缩小到 1 毫秒以内。

淘汰策略

Redis 支持以下内存淘汰策略:

  • noeviction
  • allkeys-lru
  • allkeys-random
  • volatile-lru
  • volatile-random
  • volatile-ttl

allkeys 代表针对所有 key,volatile 代表只针对带过期时间的 key

noeviction:当内存不够用时,客户端尝试执行可能需要使用更多内存的命令(除 del 和几个例外之外写命令)时,返回错误。

lru:删除最近最少使用的 key,为新数据腾出空间。

random:随机删除 key,为新数据腾出空间。

ttl:删除剩余寿命(TTL,Time To Live)最短的 key,为新数据腾出空间。

需要注意的是,volatile-lru、volatile-random、volatile-ttl 策略下,如果找不符合条件(比如没有带过期时间的 key)的 key,那么处理方式基本上就是按照 noeviction 来进行。

淘汰策略最佳实践:

  • 当你希望 key 呈幂律分布(类似二八原则,20% 的 key 承接了 80% 的访问)时,推荐使用 allkeys-lru。
  • 如果你对 key 的访问比较平均,属于雨露均沾的类型,那么推荐使用 allkeys-random。
  • 如果你的 key 大部分都是带过期时间的,那么推荐使用 volatile-ttl。

对比

过期策略

对象:带过期时间的 key

解决问题:key 到达过期时间以后,如何处理

淘汰策略

对象:所有 key(也可以只针对 expire key)

解决问题:内存不够用以后,怎么办

虽然看起来很相似,但两者干的不是一回事,它们之间没有什么直接的关系,也不会互相影响。

带你搞懂 Redis 中的两个策略相关推荐

  1. 一篇文章带你搞懂Redis(超级无敌最最最详细版本)(命令大全)(真·收藏必备)

    (本文近两万字,阅读时间可能较久,建议收藏以便查询使用) 目录 Redis诞生背景功能简介 Redis的下载与安装 Redis键的基本操作 Redis键名查询 Redis键的类型查询 Redis键的重 ...

  2. 超专业解析!10分钟带你搞懂Linux中直接I/O原理

    导语 | 本文主要以一张图为基础,向大家介绍Linux在I/O上做了哪些事情,即Linux中直接I/O原理,希望本文的经验和思路能为读者提供一些帮助和思考. 引言 我们先看一张图: 这张图大体上描述了 ...

  3. 一文带你搞懂vue中的this.$nextTick

    文章目录 1.Vue.nextTick(callback) 使用原理 2.created和mounted对比 3.例子说明 4.实际遇到的问题:vue项目中 elementUI 中表格多选框默认选中, ...

  4. 一篇文章带你搞懂Python中的类

    前言 今天我们要说的是面向对象的核心-----类,类能帮我们把复杂的事情变得有条理,有顺序,希望大家通过学习类能改善自己的编码风格,使代码变得更为好看,更加通俗易懂. 1.类的用法 一.什么是类 类( ...

  5. 一文带你搞懂 MySQL 中的分区!

    点击上方蓝色"程序猿DD",选择"设为星标" 回复"资源"获取独家整理的学习资料! 作者 | GrimMjx 来源 | https://ww ...

  6. php算法抽荣耀水晶,还在抱怨王者荣耀水晶难抽?PHP一文带你搞懂游戏中的抽奖算法...

    前言: 没有特别幸运,那么请先特别努力,别因为懒惰而失败,还矫情地将原因归于自己倒霉.你必须特别努力,才能显得毫不费力. 希望:所以说,树倒了,没有一片雪花是无辜的,抽奖都是假的,只有人家想让你中和不 ...

  7. 一篇文章带你搞懂慢SQL以及优化的策略

    文章目录 一.什么是慢SQL ? 二.为什么要对慢SQL进行优化? 三.数据库性能 1. 最大数据量 2. 最大并发数 3. 查询耗时0.5秒 4. 具体实施 四.数据库表的设计 1. 数据类型 2. ...

  8. 一篇文章带你搞懂网络层(网际层)-- 地址篇

    网络层(Network Layer)是OSI模型中的第三层(TCP/IP模型中的网际层),提供路由和寻址的功能,使两终端系统能够互连且决定最佳路径,并具有一定的拥塞控制和流量控制的能力.相当于发送邮件 ...

  9. 一文带你搞懂从动态代理实现到Spring AOP

    摘要:本文主要讲了Spring Aop动态代理实现的两种方式. 1. Spring AOP Spring是一个轻型容器,Spring整个系列的最最核心的概念当属IoC.AOP.可见AOP是Spring ...

最新文章

  1. C++ 判断某个变量是某一种类型
  2. Python 知识点笔记一常用数据类型
  3. HDUOJ---2112HDU Today
  4. Integration testing
  5. ffmpeg 源码学习之seek play
  6. pytorch 吸烟检测yolov5s
  7. PHPCMS 模板的设置
  8. 毕设/私活/兼职必备,一个挣钱的开源【SpringBoot+Spring Security+MyBatis Plus】脚手架
  9. 图片句柄_PC微信逆向:自动保存加密的聊天图片
  10. jQuery 追加元素的方法如append、prepend、before、after
  11. C# ASP.NET MVC:使用Cookie记住账号密码
  12. 干货 | 你是不是希望一月入门深度学习,三月中一篇顶会?-- 关于做科研的态度和方法的一点感想...
  13. 2. 随机变量与概率分布
  14. php中mysql数据库异步查询实现
  15. 互联网的世界安全吗?且行且珍惜
  16. 昆仑通态通用版找不到驱动_昆仑通态无法连接1200
  17. ArcGIS 几何校正
  18. final 和effectively final区别
  19. python线程isalive_如何在线程类上使用.isAlive()
  20. Transflow安装时弹出错误:You are using pip version 9.0.1, however version 19.1.1 is available. You should co

热门文章

  1. .net精美书籍大检阅
  2. 蒸汽筒行业调研报告 - 市场现状分析与发展前景预测
  3. 简单的出身年月实现 2018.9.29
  4. 七月生活20031102
  5. 业务开发工程师,你真的愿意做一辈子 CRUD boy 吗?
  6. BIOS初始化PEI至DXE阶段hob学习
  7. 失去华为后,台积电先进工艺研发疲态尽显,开始学Intel挤牙膏了
  8. 简易的java程序,银行管理系统
  9. 同事背后说坏话怎么办?为人再老实,也要做这3件事,吃亏不是福
  10. 【牛腩】-'T_news_selectByCaId' 需要参数 '@caid',但未提供该参数。”