字符串对象的编码可以是 int、raw 或 embstr
如果字符串对象保存的是整数,并且这个整数可以用 long 来表示,那么字符串对象会将整数值保存在字符串对象结构的 ptr 属性里面(将 void* 转换成 long),并将字符串对象的编码设置为 int
如果字符串对象保存的是字符串,并且这个字符串值的长度大于 39 字节,那么字符串对象将使用 SDS 来表示这个字符串值,并将对象得编码设置为 raw
如果字符串对象保存的是字符串,并且这个字符串值的长度小于等于 39 字节,那么字符串对象将使用 embstr 的方式来保存这个字符串值
embstr 编码是专门用于保存短字符串的一种优化方式,这种编码和 raw 编码一样,都使用 redisObject 结构和 SDS 结构来表示字符串对象。但 raw 编码则通过调用一次内存分配函数来分配一块连续的空间,空间中依次包含 redisObject 和 SDS 两个结构
embstr 编码的字符串对象在执行命令时,效果和 raw 编码的字符串对象执行命令时产生的效果相同,但使用 embstr 保存短字符串值有以下好处:
- embstr 编码将创建字符串对象所需的内存分配次数从 raw 编码的两次降低为一次
- 释放 embstr 编码的字符串只需要调用一次内存释放函数,而释放 raw 编码需要调用两次
- 因为 embstr 编码的字符串对象所有数据都保存在一块连续的内存中,所以比起 raw 编码的字符串对象能更好地利用缓存带来的优势
可以用 long double 类型表示的浮点数在 Redis 中也是作为字符串值来保存的。如果要保存一个浮点数到字符串对象中,程序会先将这个浮点数转换成字符串值,然后再保存转换所得的字符串值
在有需要时,程序会将保存在字符串对象里面的字符串值转换为浮点数值,执行某些操作(比如 incrbyfloat),然后将执行操作所得的浮点数值转换为字符串值,并继续保存在字符串对象中
编码的转换
int 编码和 embstr 编码的字符串对象在条件满足情况下,会被转化为 raw 编码的字符串对象
如果对 int 编码的字符串对象执行了一些命令,使这个对象不再是整数值,而是一个字符串值,那么编码将从 int 变为 raw
因为 Redis 没有为 embstr 编码的字符串对象编写任何相应的修改程序(只有 int 编码和 raw 编码的字符串对象有这些程序),所以 embstr 编码的字符串对象实际上是只读的。当对 embstr 编码的字符串对象执行任何修改命令时,程序会先将对象编码从 embstr 转换成 raw,然后再执行修改命令。因此,embstr 编码的对象再执行任何修改命令只会,总会变成一个 raw 编码的字符串对象