Redis开发与运维:SDS与44字节深入理解
对于上一篇文章,我又自己总结归纳并补充了一下,有了第二篇。
概览
<<左移
开始之前,我们先准备点东西:位运算
i<<n 总结为 i*2^n
所以
1<<5 = 2^5
1<<8 = 2^8
1<<16 = 2^16
1<<32 = 2^32
1<<64 = 2^64
SDS 5种数据类型
Redis 3.2 以后SDS数据类型有5个
#define SDS_TYPE_5 0 #define SDS_TYPE_8 1 #define SDS_TYPE_16 2 #define SDS_TYPE_32 3 #define SDS_TYPE_64 4
结合上面的位运算,我们也能理解这5个数据类型的命名规则。
外部类型String 找 SDS结构
我们现在有定义了5种SDS数据类型,那么如何根据字符串长度找这些类型呢?
或者说输入的字符串长度和类型有什么关系?下面我们来看一看他们之间的关系。
再来看看源码:
static inline char sdsReqType(size_t string_size) { if (string_size < 1<<5) return SDS_TYPE_5; if (string_size < 1<<8) return SDS_TYPE_8; if (string_size < 1<<16) return SDS_TYPE_16; #if (LONG_MAX == LLONG_MAX) if (string_size < 1ll<<32) return SDS_TYPE_32; return SDS_TYPE_64; #else return SDS_TYPE_32; #endif }
根据位运算左移公式,我可以得知 1<<8 = 2^8 = 256
那么这里的 256是指什么?这里的256就是字节
也就是说:
SDS_TYPE_5 -- 32 Byte
SDS_TYPE_8 -- 256 Byte
SDS_TYPE_16 -- 64KB
SDS_TYPE_32 -- ...
SDS_TYPE_64 -- ...
现在数据类型找到了,我们再来看看比较典型的几种操作。
追加字符串
从使用角度讲,追加一般用的频率很少。所以有多大分配多大。
所以这里追加的话,有两种大情况:还有剩余 或 不够用
主要讲一下不够用就要重新申请内存,那么我们如何去申请内存呢?
这里提供了两种分配策略:
<1M ,新空间 = 2倍扩容; >1M , 新空间 = 累加1M
空间有了,那么我们需要根据最新的空间长度占用,再找到对应的新的SDS数据类型。
看一下源码,增加一下印象:
/* 追加字符串*/ sds sdscatlen(sds s, const void *t, size_t len) { // 当前字符串长度 size_t curlen = sdslen(s); // 按需调整空间(原来字符串,要追加的长度) s = sdsMakeRoomFor(s,len); // 内存不足 if (s == NULL) return NULL; // 追加目标字符串到字节数组中 memcpy(s+curlen, t, len); // 设置追加后的长度 sdssetlen(s, curlen+len); // 追加结束符 s[curlen+len] = '\0'; return s; }
/*空间调整,注意只是调整空间,后续自己组装字符串*/ sds sdsMakeRoomFor(sds s,