分布式架构Redis中有哪些数据结构及底层实现原理


Posted in Redis onMarch 13, 2022

引言

面完了负载均衡,正向代理,反向代理,终于松了一口气,然后话题转向了缓存Redis,为什么是这个顺序呢?

回想了一下系统架构,我大概知道原因了。

分布式架构Redis中有哪些数据结构及底层实现原理

Redis 处于服务最上层。面试官是按照这个顺序从上到下考察我对整个系统设计能力,围着整个系统自顶向下的结构考察基础。不纠结这么多,反正先问后问,Redis一定是你必须掌握的。

1、面试官:我看你提到,项目中使用了Reids作为缓存,为什么是Reids而不是其他,Redis有什么优势吗?

问题分析: Redis的设计理念已经成了很多一线互联网公司自主研发分布式缓存框架的标杆,因为相比传统的 Memcache ,Redis 丰富的数据结构实在太香。

答:

  • 首先 Redis 支持丰富的数据结构,新版本数据结构从最初的5种变成9种。
  • 其次 Redis 是读写单进程单线程,不用考虑并发读写的复杂场景,速度也快。
  • Reids 功能完备,支持数据持久化,支持主从复制和集群。
  • 还有Lua脚本,事务,发布订阅模型,Reids 都支持。

在高并发请求时,为何我们频繁提到缓存技术?最直接的原因是,磁盘IO及网络开销是直接请求内存IO千百上千倍,做个简单计算,如果我们需要某个数据,该数据从数据库磁盘读出来需要0.0045S,经过网络请求传输需要0.0005S,那么每个请求完成最少需要0.005S,该数据服务器每秒最多只能响应200个请求,而如果该数据存于本机内存里,读出来只需要100us,那么每秒能够响应10000个请求。通过将数据存储到离CPU更近的未位置,减少数据传输时间,提高处理效率,这就是缓存的意义。

给您列举一个我利用Reids把项目QPS提到几十万级别的案例:

一个风控系统在日常24H中 Redis集群 QPS 曲线图,从业务低峰期几千或晚高峰最高30W,一个 Redis 集群都可轻松应对,30WQPS 在大型系统中流量并不算高,且不是核心系统,如果在多几倍几十倍多流量,一个结构优良的Redis 集群都可轻松应对,这充分说明了我们为什么要使用缓存,缓存可以把系统响应能力提高N个数量级,远高于传统基于硬盘的关系型数据库

分布式架构Redis中有哪些数据结构及底层实现原理

 面试官心想:看来是做足了功课。

2、面试官:刚刚你提到Redis是单线程,为什么单线程模型的 Redis 性能不减。

 问题分析:成功挖坑,提到单线程肯定会问我为什么要这样设计。

 答:

单线程不代表一定就慢,单线程有一个最大好处就是节省线程切换的开销,更不用考虑并发读写带来的复杂操作场景,这就大大节省了线程间切换的时间了。

单线程模型避免了多线程的频繁上下文切换,这也避免了多线程可能产生的竞争问题。

Reids 是基于内存的读写操作,内存肯定比传统磁盘IO数据库快。

Reids 核心是基于非阻塞的IO多路复用机制。

3、面试官:那你刚刚说的Redis数据结构都有哪几种,如何选择使用哪种?

问题分析: 常用的5种,重点学会这5种数据结构的使用足够了。

 答:比较常用的有5种

字符串 String: 字符串是 Redis 中最为基础的数据存储类型,数据结构简单,可存储文本,Json,图片数据等任何二进制文件。如姓名,订单号等,对于一些特殊的数据结构,比如List、Set等,建议采用相应的下面介绍的List和Set数据结构进行存储,这样不仅可以节省存储空间还可以提高操作效率。

列表 List: 类似 Java 中的 List ,按照插入顺序排序的字符串链表,在插入时,如果该键并不存在,Redis将为该键创建一个新的链表。与此相反,如果链表中所有的元素均被移除,那么该键也将会被从数据库中删除。

集合 Set: 类似 Java 中的set,但它是一个无序集合,用于存储无序(存入和取出的顺序不一定相同)元素,值不能重复。可以使用Redis的Set数据类型跟踪一些唯一性数据,比如访问系统的唯一IP地址,唯一用户ID等信息,再比如在微博应用中,每个人的好友存在一个集合(set)中,这样求两个人的共同好友的操作,可能就只需要用求交集命令即可。

有序集合 Sorted Set: 类似 Java 中的 TreeSet,支持从小到大排序的 set,适用于排行榜结构的数据存储。

Hash: 类型相当于Java中的HashMap,所以该类型非常适合于存储值对象的信息,比如用户基本信息对象含有昵称、性别和Age等属性,可以使用Hash来存储User对象,Key可以为用户的唯一ID属性。

除此之外,新版本的Redis还提供了位图,地理坐标,流几种结构。

深入分析

曾经有面试官问我,你看过Reids源码吗,我说没有看过,他说有精力可以研究一下,Redis那几种常用的数据结构底层实现原理还是值得学习的。

1、简单动态字符串结构,Redis字符串的实现方式

简单动态字符串(simple dynamic string)简称SDS。Redis使用C语言编写,但是传统的C字符串使用长度为 N+1 的字符串数组来表示长度为N的字符串,所以为了获取一个长度为C字符串的长度,必须遍历整个字符串。和C字符串不同,动态字符串的数据结构中,有专门用于保存字符串长度的变量,我们可以通过获取len属性的值,直接知道字符串长度,从一定程度上提高了读取效率。

分布式架构Redis中有哪些数据结构及底层实现原理

 Redis源码中,动态字符串的定义:

/*  
 * 保存字符串对象的结构  
 */  
struct sdshdr {  
    // buf 中已占用空间的长度  
    int len;  
    // buf 中剩余可用空间的长度  
    int free;  
    // 数据空间  
    char buf[];  
};

len 变量,用于记录buf 中已经使用的空间长度。

free 变量,用于记录buf 中还空余的空间,初次分配空间,一般没有空余,在对字符串修改的时候,会有剩余空间出现,这样做是为了杜绝C语言中缓冲区溢出的可能性,当我们需要对一个SDS进行修改的时候,Redis 会在执行拼接操作之前,预先检查给定SDS空间是否足够,如果不够,会先拓展SDS的空间,然后再执行拼接操作。

buf 字符数组,用于记录我们的字符串(记录Redis)。

2、链表数据结构,List 底层结构

链表还是常规的普通双端链表,可以支持反向查找和遍历,更方便操作,通过增删节点来灵活地调整链表的长度,双端链表在Redis内部也是被多次使用:

  • 事务模块使用双端链表依序保存输入的命令。
  • 服务器模块使用双端链表来保存多个客户端。
  • 订阅/发送模块使用双端链表来保存订阅模式的多个客户端。
  • 事件模块使用双端链表来保存时间事件(time event)。

分布式架构Redis中有哪些数据结构及底层实现原理

3、跳跃表,sorted set底层结构

 Redis sorted set的内部使用HashMap和跳跃表(SkipList)来保证数据的存储和有序,(如果你还不了解红黑树,需要先额外补补功课),HashMap里放的是成员到score的映射,而跳跃表里存放的是所有的成员,排序依据是HashMap里存的score,使用跳跃表的结构可以获得比较高的查找效率,并且在实现上比较简单。

那为什么Redis的作者使用 SkipList 结构而不是红黑树?

红黑树:红黑树的查找效率很高,但是在进行重新平衡时,会涉及到大量节点的变化,因此实现和操作起来都比较复杂。

跳跃表:通过简单的多层索引结构,实现简单,且能达到近似于红黑树的查找效率,插入节点(多层插入)不需要像红黑树那样有额外操作。而且跳跃表还能实现范围查找及输出,而红黑树只支持单个元素查找,对于范围查找效率低。

关于缓存的一些算法

(偷偷告诉你,这几个关于Reids的算法很大概率也会被问到,需要多少知道几种)

常用缓存数据淘汰策略

缓存是非常宝贵的资源,不能把所有数据都放入缓存,只能把最重要的或者要求查询速度最快的数据缓存起来,比如微博热门话题排行榜功能,通常使用缓存查询,而不是数据库。

FIFO(First In First Out): 先进先出算法,即先放入缓存的先被移除。

LRU(Least Recently Used): 最近最少使用算法,使用时间距离现在最久的那个被移除。

LFU(Least Frequently Used): 最不常用算法,一定时间段内使用次数(频率)最少的那个被移除。

缓存数据更新策略

  • 定时任务从数据库直接更新缓存:适用于对时间不敏感的数据。
  • 查询时写缓存,即查询优先查询缓存,若缓存未命中,查询数据库,将返回结果写入缓存,数据更新时先 delete缓存,再更新缓存。
  • MQ 消息异步更新缓存,后文中会针对MQ的应用做单独讲解。

总结

这一节重点讲解分布式缓存 Redis ,本地缓存不一定每个项目都会使用,但是 Redis 数据设计合理,保证超高命中率,集群足够稳定,那完全可以替代一级本地缓存。所以 Redis 非常值得你花更多时间学习。分布式缓存是面试必问。

Redis 是建设高性能网站后台不可缺少的工具,无论你是面试业务开发工程师还是架构师,都需要熟练掌握。

关于Redis,推荐阅读黄建宏的《Redis 设计与实现》,能够掌握Redis的5种数据结构,Redis 的持久化方式 RDB 和 AOF,两者有什么优点和缺点,如何选型,以及了解高可用 Redis 集群的建设方案。

以上就是分布式架构Redis中有哪些数据结构及底层实现原理的详细内容,更多关于分布式架构Redis底层数据结构及原理的资料请关注三水点靠木其它相关文章!

Redis 相关文章推荐
redis 限制内存使用大小的实现
May 08 Redis
Redis延迟队列和分布式延迟队列的简答实现
May 13 Redis
详解Redis主从复制实践
May 19 Redis
详解缓存穿透击穿雪崩解决方案
May 28 Redis
关于使用Redisson订阅数问题
Jan 18 Redis
Redis监控工具RedisInsight安装与使用
Mar 21 Redis
Redis 哨兵机制及配置实现
Mar 25 Redis
 Redis 串行生成顺序编码的方法实现
Apr 03 Redis
redis复制有可能碰到的问题汇总
Apr 03 Redis
Redis实现一个账号只能登录一个设备
Apr 19 Redis
muduo TcpServer模块源码分析
Apr 26 Redis
Redis基本数据类型Set常用操作命令
Jun 01 Redis
Redis之RedisTemplate配置方式(序列和反序列化)
Mar 13 #Redis
浅谈Redis跟MySQL的双写问题解决方案
解决linux下redis数据库overcommit_memory问题
Feb 24 #Redis
解决Redis启动警告问题
分布式Redis Cluster集群搭建与Redis基本用法
Redis命令处理过程源码解析
Redis+Lua脚本实现计数器接口防刷功能(升级版)
You might like
一个php作的文本留言本的例子(四)
2006/10/09 PHP
php类常量的使用详解
2013/06/08 PHP
php随机显示图片的简单示例
2014/02/15 PHP
阿里云PHP SMS短信服务验证码发送方法
2017/07/11 PHP
PHP 实现页面静态化的几种方法
2017/07/23 PHP
php面试实现反射注入的详细方法
2019/09/30 PHP
getElementsByTagName vs selectNodes效率 及兼容的selectNodes实现
2010/02/26 Javascript
MC Dialog js弹出层 完美兼容多浏览器(5.6更新)
2010/05/06 Javascript
可以用来调试JavaScript错误的解决方案
2010/08/07 Javascript
JavaScript限定复选框的选择个数示例代码
2013/08/25 Javascript
js有序数组的连接问题
2013/10/01 Javascript
jQuery选择器全面总结
2014/01/06 Javascript
jquery动态添加删除(tr/td)
2015/02/09 Javascript
微信小程序-图片、录音、音频播放、音乐播放、视频、文件代码实例
2016/11/22 Javascript
使用mint-ui实现省市区三级联动效果的示例代码
2018/02/09 Javascript
微信小程序下拉框搜索功能的实现方法
2019/07/31 Javascript
微信小程序 腾讯地图SDK 获取当前地址实现解析
2019/08/12 Javascript
vue中更改数组中属性,在页面中不生效的解决方法
2019/10/30 Javascript
vue+高德地图实现地图搜索及点击定位操作
2020/09/09 Javascript
[02:59]2014DOTA2西雅图国际邀请赛 圆满落幕中国夺冠
2014/07/23 DOTA
[03:08]迎霜节狂欢!2018年迎霜节珍藏Ⅰ一览
2018/12/25 DOTA
Python中splitlines()方法的使用简介
2015/05/20 Python
对Python 窗体(tkinter)树状数据(Treeview)详解
2018/10/11 Python
python 图像的离散傅立叶变换实例
2020/01/02 Python
Python接口开发实现步骤详解
2020/04/26 Python
Python如何基于Tesseract实现识别文字功能
2020/06/05 Python
Html5实现用户注册自动校验功能实例代码
2016/05/24 HTML / CSS
馥绿德雅美国官方网站:Rene Furterer头皮护理专家
2019/05/01 全球购物
经典c++面试题二
2015/08/14 面试题
Linux如何修改文件和文件夹的权限
2013/09/05 面试题
点菜员岗位职责范本
2014/02/14 职场文书
客户答谢会致辞
2015/01/20 职场文书
张丽莉观后感
2015/06/16 职场文书
小学班主任教育随笔
2015/08/15 职场文书
六五普法心得体会2016
2016/01/21 职场文书
大学生党员暑假实践(活动总结)
2019/08/21 职场文书