????相信誉过Redis的人都知道,Redis提供了一个逻辑上的对象系统构建了一个键值对数据库以供用户端客户使用。这个对象系统包括字符串对象,哈希对象,列表对象,集合对象,有序集合对象等。但是Redis面向内存并没有直接使用这些对象。而是使用了简单动态字符串,链表,字典(散列表),跳跃表,整数集合,压缩列表这些数据结构来操作内存。
Redis对象结构????Redis默认并未直接使用C字符串(C字符串仅仅作为字符串字面量,用在少量无需对字符串进行修改的地方,如打印日志)。而是以Struct的形式构造了一个SDS的笼统类型。当Redis需要一个可以被修改的字符串时,就会使用SDS来表示。在Redis数据库里,包含字符串值的键值对都是由SDS实现的(Redis中所有的键都是由字符串对象实现的即底层是由SDS实现,Redis中所有的值对象中包含的字符串对象底层也是由SDS实现)。
Redis简单动态字符串struct sdshdr{ //int 记录buf数组中未使用字节的数量 如上图free为0代表未使用字节的数量为0 int free; //int 记录buf数组中已使用字节的数量即sds的长度 如上图len为5代表未使用字节的数量为5 int len; //字节数组用于保存字符串 sds遵循了c字符串以空字符结尾的惯例目的是为了重用c字符串函数库里的函数 char buf[];}
SDS与C字符串????上图表示了SDS与C字符串的区别,关于为什么Redis要使用SDS而不是C字符串,我们可以从以下几个方面来分析。
C字符串内存溢出????C字符串,假如程序员在字符串修改的时候假如不记得给字符串重新分配足够的空间,那么就会发生内存溢出,如上图所示,不记得给s1分配足够的内存空间, s1的数据就会溢出到s2的空间, 导致s2的内容被修改.。而Redis提供的SDS其内置的空间分配策略则可以完全杜绝这种事情的发生。当API需要对SDS进行修改时, API会首先会检查SDS的空间能否满足条件, 假如不满足, API会自动对它动态扩展, 而后再进行修改。
Redis SDS字符串拼接????在C字符串中,假如对字符串进行修改,那么我们就不得不面临内存重分配。由于C字符串是由一个N+1长度的数组组成,假如字符串的长度变长,我们就必需对数组进行扩容,否则会产生内存溢出。而假如字符串长度变短,我们就必需释放掉不再使用的空间,否则会发生内存泄漏。
????对于Redis这种具备高性能要求的内存数据库,假如每次修改字符串都要进行内存重分配,无疑是巨大的性能损失。而Redis的SDS提供了两种空间分配策略来处理这个问题。
我们知道在数组进行扩容的时候,往往会申请一个更大的数组,而后把数组复制过去。为了提升性能,我们在分配空间的时候并不是分配一个刚恰好的空间,而是分配一个更大的空间。Redis同样基于这种策略提供了空间预分配。当执行字符串增长操作并且需要扩展内存时,程序不仅仅会给SDS分配必须的空间还会分配额外的未使用空间,其长度存到free属性中。其分配策略如下:
假如修改后len长度将小于1M,这时分配给free的大小和len一样,例如修改过后为10字节, 那么给free也是10字节,buf实际长度变成了10+10+1 = 21byte
假如修改后len长度将大于等于1M,这时分配给free的长度为1M,例如修改过后为30M,那么给free是1M.buf实际长度变成了30M+1M+1byte
image惰性空间释放用于字符串缩短的操作。当字符串缩短是,程序并不是立即便用内存重分配来回收缩短出来的字节,而是使用free属性记录起来,并等待将来使用。
Redis 惰性空间释放Redis通过空间预分配和惰性空间释放策略在字符串操作中肯定程度上减少了内存重分配的次数。但这种策略同样会造成肯定的内存白费,因而Redis SDS API提供相应的API让我们在有需要的时候真正的释放SDS的未使用空间。
????C字符串中的字符必需符合某种编码(比方ASCII),并且除了字符串的末尾之外,字符串里面不能包含空字符,否则最先被程序读入的空字符将被误认为是字符串结尾,这些限制使得C字符串只能保存文本数据,而不能保存像图片、音频、视频、压缩文件这样的二进制数据。假如有一种使用空字符来分割多个单词的特殊数据格式,就不能用C字符串来表示,如"Redis\0String",C字符串的函数会把'\0'当做结束符来解决,而忽略到后面的"String"。而SDS的buf字节数组不是在保存字符,而是一系列二进制数组,SDS API都会以二进制的方式来解决buf数组里的数据,使用len属性的值而不是空字符来判断字符串能否结束。
????我们来看几个Redis常见操作的时间复杂度。
获取SDS长度: 因为SDS中提供了len属性,因而我们可以直接获取时间复杂度为O(1),C字符串为O(n)。
获取SDS未使用空间长度: 时间复杂度为0(1),起因同1。
清理SDS保存的内容:因为惰性空间分配策略,复杂度为O(1)。
创立一个长度为N的字符串:时间复杂度为O(n)。
拼接一个长度为N的C字符串:时间复杂度为O(n)。
拼接一个长度为N的SDS字符串:时间复杂度为O(n)。
Redis在获取字符串长度上的时间复杂度为常数级O(1)。
????通过以上分析,我们可以得到,SDS这种数据结构相对于C字符串有以下优点:
杜绝缓冲区溢出
减少字符串操作中的内存重分配次数
二进制安全
因为SDS遵循以空字符结尾的惯例,因而兼容部门C字符串函数
Redis定位于一个高性能的内存数据库,其面向的就是大数据量,大并发,频繁读写,高响应速度的业务。因而在保证安全稳固的情况下,性能的提升非常重要。而SDS这种数据结构屏蔽了C字符串的少量缺点,可以提供安全高性能的字符串操作。
????Redis在互联网项目中的应用越来越广泛,会用只是学习Redis中最简单的一步,要想真正的成为Redis高手,理解其底层的实现必不可少。本篇文章简单详情了Redis中SDS数据结构及其特性,分析了Redis SDS的空间分配策略和其与C字符串相比的优势,后续的文章将继续分享Redis底层实现的其它数据结构。未完待续......
《Redis设计与实现》
《Redis开发与运维》
《Redis官方文档》
¥23.00
正版 steam 为了吾王 For The King 为了国王 国区激活码 cdkey 正版简体中文 PC中文游戏
¥78.00
uplay孤岛惊魂6 孤岛6 FarCry6 FC6 远哭6游戏激活码育碧激活码正版PC季票豪华黄金终极DLC自动发货
¥75.00
女神异闻录4黄金版 女神异闻录4steam 无敌究极后桥背摔 游戏中文国区激活码cdkey
¥99.00
漫威暗夜之子 Marvel's Midnight Suns steam PC中文游戏 激活码KEY
¥85.00
PC中文正版游戏STEAM 人类 HUMANKIND 国区激活码
¥34.50
PC中文正版 Steam平台游戏 代号探戈 Operation Tango 代号:探戈 国区激活码 联机合作