redis之简单动态字符串SDS

1、SDS简介

    无论是 Redis 的 Key 还是 Value,其基础数据类型都是字符串。例如,Hash 型 Value 的 field 与 value 的类型、List 型、Set 型、ZSet 型 Value 的元素的类型等都是字符串。虽然 Redis 是使用标准 C 语言开发的,但并没有直接使用 C 语言中传统的字符串表示,而是自定义了一 种字符串。这种字符串本身的结构比较简单,但功能却非常强大,称为简单动态字符串, Simple Dynamic String,简称 SDS。

      注意,Redis 中的所有字符串并不都是 SDS,也会出现 C 字符串。C 字符串只会出现在字 符串“字面常量”中,并且该字符串不可能发生变更。 redisLog(REDIS_WARNNING, “sdfsfsafsafds”);

 2、SDS结构

   SDS 不同于 C 字符串。C 字符串本身是一个以双引号括起来,以空字符’\0’结尾的字符序列。但SDS是一个结构体,定义在Redis安装目录下的src/sds.h中。

struct sdshdr {
// 字节数组,用于保存字符串
char buf[];
// buf[]中已使用字节数量,称为 SDS 的长度
int len;
// buf[]中尚未使用的字节数量
int free;
}

例如执行 SET country “China”命令时,键 country 与值”China”都是 SDS 类型的,只不过 一个是 SDS 的变量,一个是 SDS 的字面常量。”China”在内存中的结构如下:

通过以上结构可以看出,SDS 的 buf 值实际是一个 C 字符串,包含空字符’\0’共占 6 个字 节。但 SDS 的 len 是不包含空字符’\0’的。

该结构与前面不同的是,这里有 3 字节未使用空间。

3、SDS的优点

     C 字符串使用 Len+1 长度的字符数组来表示实际长度为 Len 的字符串,字符数组最后以 空字符’\0’结尾,表示字符串结束。这种结构简单,但不能满足 Redis 对字符串功能性、安全 性及高效性等的要求。

(1)、防止“字符串长度获取”性能瓶颈

    对于C字符串,若要获取其长度,则必须要通过遍历整个字符串才可以获取到的。对于超常字符串的遍历,会成为系统的性能瓶颈。

       但是,由于SDS结构体中直接就存放着字符串的长度数据,所以对于获取字符串长度需要消耗的系统性能,与字符串本身长度是无关的,不会成为Redis的性能瓶颈。

(2)保障二进制安全

      C 字符串中只能包含符合某种编码格式的字符,例如 ASCII、UTF-8 等,并且除了字符串 末尾外,其它位置是不能包含空字符’\0’的,否则该字符串就会被程序误解为提前结束。而 在图片、音频、视频、压缩文件、office 文件等二进制数据中以空字符’\0’作为分隔符的情况 是很常见的。故而在 C 字符串中是不能保存像图片、音频、视频、压缩文件、office 文件等 二进制数据的。 但 SDS 不是以空字符’\0’作为字符串结束标志的,其是通过 len 属性来判断字符串是否 结束的。所以,对于程序处理 SDS 中的字符串数据,无需对数据做任何限制、过滤、假设, 只需读取即可。数据写入的是什么,读到的就是什么。

(3)、减少内存再分配次数

    SDS采用了空间预分配策略惰性空间释放策略来避免内存再分配问题。

    空间预分配策略是指,每次SDS进行空间扩展时,程序不但为其分配所需的空间,还会为其分配额外的未使用的空间,以减少内存再分配次数。而额外分配的未使用空间大小取决于空间扩展后SDS的len属性值。

   如果 len 属性值小于 1M,那么分配的未使用空间 free 的大小与 len 属性值相同。

   如果 len 属性值大于等于 1M ,那么分配的未使用空间 free 的大小固定是 1M。

   SDS对于空间释放采用的是惰性空间释放策略。该策略是指,SDS字符串长度如果缩短,那么多出的未使用空间将暂时不释放,而是增加到free中。以使后期扩展SDS时减少内存再分配次数。

   如果要释放 SDS 的未使用空间,则可通过 sdsRemoveFreeSpace()函数来释放。

(4)兼容C函数

   Redis 中提供了很多的 SDS 的 API,以方便用户对 Redis 进行二次开发。为了能够兼容 C 函数,SDS 的底层数组 buf[]中的字符串仍以空字符’\0’结尾。 现在要比较的双方,一个是 SDS,一个是 C 字符串,此时可以通过 C 语言函数 strcmp(sds_str->buf,c_str)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值