带你搞懂 Redis 中的两个策略
面试的时候问到候选人 Redis 相关问题时,发现一个现象:一部分候选人分不清 Redis 的「键过期策略」和「内存淘汰策略」。今天就来说一说这老哥俩。
简单来说,过期策略就是当 key 到了指定的过期时间后,Redis 是用什么方式将其删除的;而淘汰策略指的是当内存不够用时,Redis 如何处理。
过期策略
Redis 的键过期方式有两种:被动和主动。
**被动方式:**当客户端尝试访问某个过期 key 时,Redis 发现该 key 已过期,将其删除。
**主动方式:**为每个设置了过期时间的 key 设置一个定时器,当到达过期时间时将 key 删除。
被动方式的问题很明显,可能有些 key 很久不会被访问,甚至永远都不会被访问,浪费内存资源。
主动方式也有问题,需要不断的去查看 key 是否已经过期,过于消耗 CPU 资源。
还有一种定期删除的方式,比如 30s 对设置过期时间的 key 进行一次扫描,并删除那些已经过期的 key。这种方式看起来是结合前两者的优势,但是它还是过于粗暴。
那么,Redis 到底是如何处理过期 key 的呢?答案是,被动 + 优化版的定期删除。被动已经说过了,下面看看 Redis 是怎么优化定期删除的。
Redis 是如何做的:
- 随机选择 20 个设置带有过期时间的 key
- 删除其中已经到达过期时间的 key
- 如果 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 中的两个策略相关推荐
- 一篇文章带你搞懂Redis(超级无敌最最最详细版本)(命令大全)(真·收藏必备)
(本文近两万字,阅读时间可能较久,建议收藏以便查询使用) 目录 Redis诞生背景功能简介 Redis的下载与安装 Redis键的基本操作 Redis键名查询 Redis键的类型查询 Redis键的重 ...
- 超专业解析!10分钟带你搞懂Linux中直接I/O原理
导语 | 本文主要以一张图为基础,向大家介绍Linux在I/O上做了哪些事情,即Linux中直接I/O原理,希望本文的经验和思路能为读者提供一些帮助和思考. 引言 我们先看一张图: 这张图大体上描述了 ...
- 一文带你搞懂vue中的this.$nextTick
文章目录 1.Vue.nextTick(callback) 使用原理 2.created和mounted对比 3.例子说明 4.实际遇到的问题:vue项目中 elementUI 中表格多选框默认选中, ...
- 一篇文章带你搞懂Python中的类
前言 今天我们要说的是面向对象的核心-----类,类能帮我们把复杂的事情变得有条理,有顺序,希望大家通过学习类能改善自己的编码风格,使代码变得更为好看,更加通俗易懂. 1.类的用法 一.什么是类 类( ...
- 一文带你搞懂 MySQL 中的分区!
点击上方蓝色"程序猿DD",选择"设为星标" 回复"资源"获取独家整理的学习资料! 作者 | GrimMjx 来源 | https://ww ...
- php算法抽荣耀水晶,还在抱怨王者荣耀水晶难抽?PHP一文带你搞懂游戏中的抽奖算法...
前言: 没有特别幸运,那么请先特别努力,别因为懒惰而失败,还矫情地将原因归于自己倒霉.你必须特别努力,才能显得毫不费力. 希望:所以说,树倒了,没有一片雪花是无辜的,抽奖都是假的,只有人家想让你中和不 ...
- 一篇文章带你搞懂慢SQL以及优化的策略
文章目录 一.什么是慢SQL ? 二.为什么要对慢SQL进行优化? 三.数据库性能 1. 最大数据量 2. 最大并发数 3. 查询耗时0.5秒 4. 具体实施 四.数据库表的设计 1. 数据类型 2. ...
- 一篇文章带你搞懂网络层(网际层)-- 地址篇
网络层(Network Layer)是OSI模型中的第三层(TCP/IP模型中的网际层),提供路由和寻址的功能,使两终端系统能够互连且决定最佳路径,并具有一定的拥塞控制和流量控制的能力.相当于发送邮件 ...
- 一文带你搞懂从动态代理实现到Spring AOP
摘要:本文主要讲了Spring Aop动态代理实现的两种方式. 1. Spring AOP Spring是一个轻型容器,Spring整个系列的最最核心的概念当属IoC.AOP.可见AOP是Spring ...
最新文章
- C++ 判断某个变量是某一种类型
- Python 知识点笔记一常用数据类型
- HDUOJ---2112HDU Today
- Integration testing
- ffmpeg 源码学习之seek play
- pytorch 吸烟检测yolov5s
- PHPCMS 模板的设置
- 毕设/私活/兼职必备,一个挣钱的开源【SpringBoot+Spring Security+MyBatis Plus】脚手架
- 图片句柄_PC微信逆向:自动保存加密的聊天图片
- jQuery 追加元素的方法如append、prepend、before、after
- C# ASP.NET MVC:使用Cookie记住账号密码
- 干货 | 你是不是希望一月入门深度学习,三月中一篇顶会?-- 关于做科研的态度和方法的一点感想...
- 2. 随机变量与概率分布
- php中mysql数据库异步查询实现
- 互联网的世界安全吗?且行且珍惜
- 昆仑通态通用版找不到驱动_昆仑通态无法连接1200
- ArcGIS 几何校正
- final 和effectively final区别
- python线程isalive_如何在线程类上使用.isAlive()
- Transflow安装时弹出错误:You are using pip version 9.0.1, however version 19.1.1 is available. You should co