接下来咱们来一块儿研究下Redis工程架构相关的问题,这部份内容出现的几率相对大一些,由于并非全部人都会去研究源码,若是面试一味问源码那么可能注定是一场尬聊。git
面试时在不要求候选人对Redis很是熟练的前提下,工程问题将是不二之选。github
经过本文你将了解到如下内容:
1.Redis的内存回收详解
2.Redis的持久化机制面试
1.1 为何要回收内存?redis
Redis做为内存型数据库,若是单纯的只进不出迟早就撑爆了,事实上不少把Redis当作主存储DB用的家伙们迟早会尝到这个苦果,固然除非你家厂子确实不差钱,数T级别的内存都毛毛雨,或者数据增加必定程度以后再也不增加的场景,就另当别论了。算法
对于咱们这种把节约成本当作KPI的普通厂子,仍是把Redis当缓存用比较符合家里的经济条件,因此这么看面试官的问题还算是比较贴合实际,比起那些手撕RBTree好一些,若是问题恰好在你知识射程范围内,先给面试官点个赞再说!数据库
为了让Redis服务安全稳定的运行,让使用内存保持在必定的阈值内是很是有必要的,所以咱们就须要删除该删除的,清理该清理的,把内存留给须要的键值对,试想一条大河须要设置几个警惕水位来确保不决堤不枯竭,Redis也是同样的,只不过Redis只关心决堤便可,来一张图:缓存
图中设定机器内存为128GB,占用64GB算是比较安全的水平,若是内存接近80%也就是100GB左右,那么认为Redis目前承载能力已经比较大了,具体的比例能够根据公司和我的的业务经验来肯定。安全
笔者只是想表达出于安全和稳定的考虑,不要以为128GB的内存就意味着存储128GB的数据,都是要打折的。架构
1.2 内存从哪里回收?dom
Redis占用的内存是分为两部分:存储键值对消耗和自己运行消耗。显而后者咱们没法回收,所以只能从键值对下手了,键值对能够分为几种:带过时的、不带过时的、热点数据、冷数据。对于带过时的键值是须要删除的,若是删除了全部的过时键值对以后内存仍然不足怎么办?那只能把部分数据给踢掉了。
人生无处不取舍,这个让笔者脑海浮现了《泰坦尼克》,邮轮撞到了冰山顷刻间海水涌入,面临数量不足的救生艇,人们作出了抉择:让女士和孩童先走,绅士们选择留下,海上逃生场景如图:
要实施对键值对的删除咱们须要明白以下几点:
老规矩来到github看下源码,src/server.h中给的redisDb结构体给出了答案:
typedef struct redisDb { dict *dict; /* The keyspace for this DB */ dict *expires; /* Timeout of keys with a timeout set */ dict *blocking_keys; /* Keys with clients waiting for data (BLPOP)*/ dict *ready_keys; /* Blocked keys that received a PUSH */ dict *watched_keys; /* WATCHED keys for MULTI/EXEC CAS */ int id; /* Database ID */ long long avg_ttl; /* Average TTL, just for stats */ unsigned long expires_cursor; /* Cursor of the active expire cycle. */ list *defrag_later; /* List of key names to attempt to defrag one by one, gradually. */ } redisDb;
Redis本质上就是一个大的key-value,key就是字符串,value有是几种对象:字符串、列表、有序列表、集合、哈希等,这些key-value都是存储在redisDb的dict中的,来看下黄健宏画的一张很是赞的图:
看到这里,对于删除机制又清晰了一步,咱们只要把redisDb中dict中的目标key-value删掉就行,不过貌似没有这么简单,Redis对于过时键值对确定有本身的组织规则,让咱们继续研究吧!
redisDb的expires成员的类型也是dict,和键值对是同样的,本质上expires是dict的子集,expires保存的是全部带过时的键值对,称之为过时字典吧,它才是咱们研究的重点。
对于键,咱们能够设置绝对和相对过时时间、以及查看剩余时间:
上述三组命令在设计缓存时用处比较大,有心的读者能够留意。
过时字典expires和键值对空间dict存储的内容并不彻底同样,过时字典expires的key是指向Redis对应对象的指针,其value是long long型的unix时间戳,前面的EXPIRE和PEXPIRE相对时长最终也会转换为时间戳,来看下过时字典expires的结构,笔者画了个图:
判断键是否过时可删除,须要先查过时字典是否存在该值,若是存在则进一步判断过时时间戳和当前时间戳的相对大小,作出删除判断,简单的流程如图:
通过前面的几个环节,咱们知道了Redis的两种存储位置:键空间和过时字典,以及过时字典expires的结构、判断是否过时的方法,那么该如何实施删除呢?
先抛开Redis来想一下可能的几种删除策略:
在上述的三种策略中定时删除和按期删除属于不一样时间粒度的主动删除,惰性删除属于被动删除。
三种策略都有各自的优缺点:
定时删除对内存使用率有优点,可是对CPU不友好,惰性删除对内存不友好,若是某些键值对一直不被使用,那么会形成必定量的内存浪费,按期删除是定时删除和惰性删除的折中。
Reids采用的是惰性删除和定时删除的结合,通常来讲能够借助最小堆来实现定时器,不过Redis的设计考虑到时间事件的有限种类和数量,使用了无序链表存储时间事件,这样若是在此基础上实现定时删除,就意味着O(N)遍历获取最近须要删除的数据。
可是我以为antirez若是非要使用定时删除,那么他确定不会使用原来的无序链表机制,因此我的认为已存在的无序链表不能做为Redis不使用定时删除的根本理由,冒昧猜想惟一可能的是antirez以为没有必要使用定时删除。
按期删除听着很简单,可是如何控制执行的频率和时长呢?
试想一下若是执行频率太少就退化为惰性删除了,若是执行时间太长又和定时删除相似了,想一想还确实是个难题!而且执行按期删除的时机也须要考虑,因此咱们继续来看看Redis是如何实现按期删除的吧!笔者在src/expire.c文件中找到了activeExpireCycle函数,按期删除就是由此函数实现的,在代码中antirez作了比较详尽的注释,不过都是英文的,试着读了一下模模糊糊弄个大概,因此学习英文并阅读外文资料是很重要的学习途径。
因为笔者对Redis源码了解很少,只能作个模糊版本的解读,因此不免有问题,仍是建议有条件的读者自行前往源码区阅读,抛砖引玉看下笔者的模糊版本:
主体意思:按期删除是个自适应的闭环而且几率化的抽样扫描过程,过程当中都有执行时间和cpu时间的限制,若是触发阈值就中止,能够说是尽可能在不影响对客户端的响应下润物细无声地进行的。
1.3.5 DEL删除键值对
在Redis4.0以前执行del操做时若是key-value很大,那么可能致使阻塞,在新版本中引入了BIO线程以及一些新的命令,实现了del的延时懒删除,最后会有BIO线程来实现内存的清理回收。
为了保证Redis的安全稳定运行,设置了一个max-memory的阈值,那么当内存用量到达阈值,新写入的键值对没法写入,此时就须要内存淘汰机制,在Redis的配置中有几种淘汰策略能够选择,详细以下:
后三种策略都是针对过时字典的处理,可是在过时字典为空时会noeviction同样返回写入失败,毫无策略地随机删除也不太可取,因此通常选择第二种allkeys-lru基于LRU策略进行淘汰。
我的认为antirez一贯都是工程化思惟,善于使用几率化设计来作近似实现,LRU算法也不例外,Redis中实现了近似LRU算法,而且通过几个版本的迭代效果已经比较接近理论LRU算法的效果了,这个也是个不错的内容,因为篇幅限制,本文计划后续单独讲LRU算法时再进行详细讨论。
过时健删除策略强调的是对过时健的操做,若是有健过时而内存足够,Redis不会使用内存淘汰机制来腾退空间,这时会优先使用过时健删除策略删除过时健。
内存淘汰机制强调的是对内存数据的淘汰操做,当内存不足时,即便有的健没有到达过时时间或者根本没有设置过时也要根据必定的策略来删除一部分,腾退空间保证新数据的写入。
Q2:讲讲你对Redis持久化机制的理解。
我的认为Redis持久化既是数据库自己的亮点,也是面试的热点,主要考察的方向包括:RDB机制原理、AOF机制原理、各自的优缺点、工程上的对于RDB和AOF的取舍、新版本Redis混合持久化策略等,如能把握要点,持久化问题就过关了。