Redis面试题

1. Redis是单线程还是多线程?

ans: 最新的Redis采用并发安全的单线程的数据持久化,和高效率的多线程的网络请求和数据操作。

Redis6.0以前是单线程。这的单线程是说其网络I/O和键值对读写是由一个线程完成的。所以Redis是并发安全的。
Redis6.0引入了多线程,是对网络请求过程采用了多线程(利用CPU的多核实现),而其键值对读写依然采用并发安全的单线程处理。
所以Redis依然是并发安全的。

也就是说只有网络请求模块和数据操作模块是单线程的,而其数据持久化和集群数据同步是由额外的线程执行的。

                                ---request--->                                <---request---
Redis客户端01                                   Redis服务器                                    Redis客户端02

                                <---response---                              ---response--->    

                                                      

2. Redis单线程为什么还能这么快?

ans: 采用高效数据存储结构,基于内存操作,命令单线程操作,基于高效的I/O多路复用机制。

(1)Redis基于纯内存操作,一条命令在内存里操作时间是几十纳秒

(2)Redis命令的执行是单线程操作,没有线程切换开销

(3)Redis基于I/O多路复用机制来提升I/O利用率

(4)Redis具有高效的数据存储结构:  全局Hash表,以及跳表,压缩列表,链表等多种高效数据结构。

Hash表由一维的数组和二维的链表组成,

时间复杂度是一个常量O(1),是一种效率相当高的一种存储结构。

3. Redis底层数据是如何用跳表来存储的?

ans: 跳表是类似于折半查找的优化的链表

跳表: 由有序链表优化而来,每两个元素使用第一个元素作为冗余元素,索引元素。

索引元素也每两个找第一个作为索引,直到生成第一层是第一个元素的唯一索引

从跳表最左侧的冗余元素开始查。

类似于折半查找。

4. Redis的关键字key过期了为什么没有释放

ans: 修改未设置过期时间导致过期时间丢失

使用Redis时常使用SET命令,

SET命令除了可以设置键值对key-value以外,还可以设置key的过期时间:

> SET fhe kiri lulu EX 120

OK

>TTL fhe

117

此时如果你想修改key的值,但是只是单纯使用SET命令,而没有加上过期时间参数,那么这个key的过期时间就会被擦除,也就是这个key永远不过期了。

>SET fhe fhe123
OK
> TTL fhe
-1    // 改值不设置过期时间,那么这个key永远不会过期了!

所以,导致这个问题的原因在于:SET命令如果不设置过期时间,那么Redis会默认擦除该key的工期时间。

如果你发现Redis的内存持续增长,而且很多key原来设置了过期时间,后来发现过期时间丢失,就很可能是修改时没有设置过期时间导致其过期时间被擦除。

这时,你的Redis中就会存在大量不过期的key,从而会消耗过多的内存资源。

所以你在使用SET命令时,如果刚开始就设置了过期时间,那么之后修改了这个key,则也要务必加上过期时间的参数,避免过期时间丢失问题。