🎈🎈在.NET后端开发岗位中,如今也少不了、微服务、分布式、高并发高可用相关的面试题🎈🎈
👍👍本文分享一些整理的Redis高频面试题🎉
👍👍机会都是给有准备的人的,祝你一面而就🎉
Redis 是一个基于内存的高性能key-value数据库。Redis优势:
性能极高 – Redis以内存作为数据存储介质,读的速度是110000次/s,写的速度是81000次/s 。Redis高性能是因为:1. 纯内存操作 2.单线程操作,避免了频繁的上下文切换 3.采用了非阻塞 I/O 多路复用机制
丰富的数据类型 – Redis支持二进制案例的 Strings, Lists, Hashes, Sets 及 Ordered Sets 数据类型操作。
原子操作 – Redis的所有操作都是原子性的,同时Redis还支持对几个操作合并后的原子性执行(事务)。
丰富的特性 – Redis还支持发布/订阅(publish/subscribe)、通知、键值(Key)过期等等特性。
Redis的缺点也必须了解:
Redis跟普通的内存缓存不同的是,它有持久化机制,可以根据持久化策略把内存中的数据备份到磁盘中。宕机重启后,Redis服务器程序会把磁盘中的数据再重新写到内存中从而恢复数据。
第4步:操作系统将缓冲区中的数据转移到磁盘控制器上(数据在磁盘缓存中)。
有两种策略:RDB和AOF,可以在redis.conf的配置文件中设置。
AOF持久化:工作机制很简单,Redis会将每一个收到的写命令都通过write函数追加到文件中。通俗的理解就是“日志记录”。AOF记录服务器的所有写操作,在服务器重新启动的时候,会把所有的写操作重新执行一遍,从而实现数据备份。
AOF存在的问题及解决方案:AOF的方式也同时带来了另一个问题。持久化文件会变的越来越大。为了压缩AOF的持久化文件,Redis提供了bg rewrite aof命令。将内存中的数据以命令的方式保存到临时文件中,同时会fork出一条新进程来将文件重写。
重写aof文件的操作,并没有读取旧的aof文件,而是将整个内存中的数据库内容用命令的方式重写了一个新的aof文件,这点和快照有点类似。(以上文件重写机制要着重了解一下)
RDB持久化:RDB持久化是指在指定的时间间隔内将内存中的数据集快照写入磁盘,是一种全量备份方式。也是默认的持久化方式,这种方式是就是将内存中数据以快照的方式写入到二进制文件中,默认的文件名为dump.rdb。服务器启动的时候,可以从 RDB 文件中恢复数据集。
两种持久化方式怎么选?需求不同选择不同,但是通常都是结合使用。两者的对比看下图:
相同点:Redis和本地缓存(MemoryCache)都是将数据存放在内存中,都是内存数据库。
数据类型:Redis不仅支持简单的k/v类型的数据(string类型),同时还提供list,set,hash等数据结构的存储;本地缓存没有那么丰富的数据类型,通常仅支持简单k/v类型数据。
持久化功能:Redis有数据持久化功能,而本地缓存没有持久化功能;
应用场景不一样:Redis除了作为NoSQL数据库使用外,还能用做消息队列、数据堆栈和数据缓存等;本地缓存适合于缓存SQL语句、数据集、用户临时性数据、延迟查询数据和session等。
三种集群模式:主从模式、哨兵模式、Cluster集群模式。
?
MySql是关系型数据库,主要用于存放持久化数据,将数据存储在硬盘中,读取速度较慢。
Sql Server是关系型数据库,读取速度较慢、只能运行在windows平台,难处理高并发等性能问题。
Redis是NoSQL即非关系型数据库,也是缓存数据库,即将数据存储在缓存中,缓存的读取速度快,能够大大的提高运行效率,但是保存时间有限。
缓存穿透是指用户查询数据,在数据库没有,自然在缓存中也不会有。这样就导致用户查询的时候,在缓存中找不到对应key的value,每次都要去数据库再查询一遍,然后返回空(相当于进行了两次无用的查询)。这样请求就绕过缓存直接查数据库。
解决方案:
采用布隆过滤器(Bloom Filter):在缓存之前在加一层BloomFilter,在查询的时候先去BloomFilter去查询key是否存在,如果不存在就直接返回,存在再去查询缓存,缓存中没有再去查询数据库。
如果缓存集中在一段时间内失效,发生大量的缓存穿透,所有的查询都落在数据库上,造成了缓存雪崩。由于原有缓存失效,新缓存未到期间,所有原本应该访问缓存的请求都去查询数据库了,而对数据库CPU和内存造成巨大压力,严重的会造成数据库宕机。
解决方案:
在高并发系统中,大量的请求同时查询一个key时,此时这个key正好失效了,就会导致大量的请求都打到数据库上面去。这种现象我们称为缓存击穿。这个和缓存雪崩的区别在于这里针对某一key缓存,前者则是很多key。
解决方案:
缓存预热就是系统上线后,将相关的缓存数据直接加载到缓存系统。这样就可以避免在用户请求的时候,先查询数据库,然后再将数据缓存的问题。用户直接查询事先被预热的缓存数据。
缓存预热的不同策略:
1)直接删除redis缓存
2)基于消息队列(MQ)形式实现同步
3)基于canal订阅MySql的binlog二进制文件,通过mq实现异步同步
未完待续...