关注
Q4:
我们都知道Redis对象可以实现一种类型不同底层实现,我们现在来看字符串对象。你可以简单地说一下它的几种底层实现方案?然后讲一下为什么embstr和raw两种实现方案里要以字符串39个字节为界限或者说3版本往后为什么要使用44个字节为界限?
A:
在Redis里,字符串是一个对象,对象是由一个RedisObject结构表示,这个结构中和保存数据有关的三个属性分别是`type`,`encoding`,`ptr`,encoding对应每种类型的不同底层实现方式。
字符串的三种底层实现方案分别对应不同的应用场景,当value的值是数字时,会将指向底层实现的ptr指针变换为long类型,然后将数字直接存放进去,同时将encoding改为int;当value是一个字符串,且大小低于44个字节时(3版本用的是39个字节),采用类似压缩列表的做法,将SDS直接跟在redisObject后面,而不是通过指针指向,同时将编码改为embstr;当这个字符串大于44个字节时(3版本以前用的是39字节),使用SDS实现,并且由指针指向SDS的地址,将编码改为raw。
而使用39个字节作为分类标准的原因是因为计算机分配内存时通常以8为界限,其中8,16,32,64为分配单位,而最开始的RedisObject的头已经占据16个字节,而SDS对象的头又会占据8个字节(两个unsigned int会占据8个字节),因此留给字符串的空间只有40个字节,又因为字符串的最后一个字符默认为`\0`,因此剩下39个字节。而新版本以44为界限的原因是,每个sds都有一个sdshdr,里面的len和free记录了这个sds的长度和空闲空间,但是这样的处理十分粗糙,使用的unsigned int可以表示很大的范围,但是对于很短的sds有很多的空间被浪费了(两个unsigned int 8个字节)。新版本将原来的sdshdr改成了sdshdr8,sdshdr16,sdshdr32,sdshdr64,里面的unsigned int 变成了uint8_t,uint16_t,这样更加优化小sds的内存使用;因此最小的字符串对象头仅仅占用3个字节(len,alloc都是1个字节,还有一个char字符flag占用一个字节),因此节约了5个字节的空间,留出来的就是44个字节。
查看原帖
点赞 评论
相关推荐
点赞 评论 收藏
分享
牛客热帖
正在热议
# 拼多多求职进展汇总 #
233206次浏览 2030人参与
# 在职场上,你最讨厌什么样的同事 #
5704次浏览 81人参与
# 阿里云管培生offer #
58873次浏览 1748人参与
# 25届秋招总结 #
396126次浏览 3975人参与
# 哪些公司校招卡第一学历 #
32815次浏览 105人参与
# 地方国企笔面经互助 #
6524次浏览 16人参与
# 北方华创开奖 #
65977次浏览 549人参与
# ai智能作图 #
21225次浏览 261人参与
# 硬件兄弟们 甩出你的华为奖状 #
77931次浏览 625人参与
# 实习,投递多份简历没人回复怎么办 #
2435521次浏览 34701人参与
# 工作中,你有没有遇到非常爱骂人的领导? #
4714次浏览 47人参与
# 实习与准备秋招该如何平衡 #
722713次浏览 8551人参与
# 我的实习求职记录 #
6120621次浏览 83950人参与
# 如果再来一次,你还会选择这个工作吗? #
110289次浏览 1109人参与
# 25届机械人为了秋招做了哪些准备? #
24979次浏览 355人参与
# 签了三方后想毁约怎么办 #
18554次浏览 111人参与
# 如果你有一天可以担任公司的CEO,你会做哪三件事? #
9940次浏览 213人参与
# 机械求职避坑tips #
22142次浏览 240人参与
# 游戏求职进展汇总 #
52729次浏览 344人参与
# 夸夸我的求职搭子 #
132006次浏览 1360人参与
# 腾讯求职进展汇总 #
207522次浏览 1694人参与
# 实习想申请秋招offer,能不能argue薪资 #
35771次浏览 308人参与