2024年【redis源码学习】redisObject,互联网java工程师面试题

读者福利

秋招我借这份PDF的复习思路,收获美团,小米,京东等Java岗offer

更多笔记分享

秋招我借这份PDF的复习思路,收获美团,小米,京东等Java岗offer

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

robj


redis中的数据对象 server.h/redisObject 是redis内部存储的数据定义的抽象类型。

//英文是自带的,中文是我写的

typedef struct redisObject {

unsigned type:4; //数据类型

unsigned encoding:4;//编码格式

unsigned lru:LRU_BITS; /* LRU time (relative to global lru_clock) or

  • LFU data (least significant 8 bits frequency

  • and most significant 16 bits access time). */

//LRU时间戳 or LRU计数

int refcount; //引用计数,为了节省内存,redis会在多处引用同一个redisObject

void *ptr; //指向实际的数据结构

} robj;

前面仨儿,嗯,挤一个 unsigned 的不同地址位。嗯,大师果然是艰苦朴素,勤俭持“内存”啊。


数据类型


/* A redis object, that is a type able to hold a string / list / set */

/* The actual Redis Object */

#define OBJ_STRING 0 /* String object. */

#define OBJ_LIST 1 /* List object. */

#define OBJ_SET 2 /* Set object. */

#define OBJ_ZSET 3 /* Sorted set object. */

#define OBJ_HASH 4 /* Hash object. */

/* The “module” object type is a special one that signals that the object

  • is one directly managed by a Redis module. In this case the value points

  • to a moduleValue struct, which contains the object value (which is only

  • handled by the module itself) and the RedisModuleType struct which lists

  • function pointers in order to serialize, deserialize, AOF-rewrite and

  • free the object.

  • Inside the RDB file, module types are encoded as OBJ_MODULE followed

  • by a 64 bit module type ID, which has a 54 bits module-specific signature

  • in order to dispatch the loading to the right module, plus a 10 bits

  • encoding version. */

#define OBJ_MODULE 5 /* Module object. 自定义消息类型*/

#define OBJ_STREAM 6 /* Stream object. 消息流*/


编码类型


/* Objects encoding. Some kind of objects like Strings and Hashes can be

  • internally represented in multiple ways. The ‘encoding’ field of the object

  • is set to one of this fields for this object. */

#define OBJ_ENCODING_RAW 0 /* Raw representation 简单动态字符串*/

#define OBJ_ENCODING_INT 1 /* Encoded as integer 整数*/

#define OBJ_ENCODING_HT 2 /* Encoded as hash table 字典*/

#define OBJ_ENCODING_ZIPMAP 3 /* Encoded as zipmap 未使用*/

#define OBJ_ENCODING_LINKEDLIST 4 /* No longer used: old list encoding. 不再使用*/

#define OBJ_ENCODING_ZIPLIST 5 /* Encoded as ziplist 压缩列表*/

#define OBJ_ENCODING_INTSET 6 /* Encoded as intset 整数集合*/

#define OBJ_ENCODING_SKIPLIST 7 /* Encoded as skiplist 跳表*/

#define OBJ_ENCODING_EMBSTR 8 /* Embedded sds string encoding 简单动态字符串*/

#define OBJ_ENCODING_QUICKLIST 9 /* Encoded as linked list of ziplists 快速链表*/

#define OBJ_ENCODING_STREAM 10 /* Encoded as a radix tree of listpacks 流*/


随机应变的对象编码


对象的整个周期中,编码不是一成不变的。也是为了节约嘛。比如上面可以看到有个整数集合,当集合中所有元素都可以用整数表示时,底层数据结构采用整数集合。看:

int setTypeAdd(robj *subject, sds value) {

long long llval;

if (subject->encoding == OBJ_ENCODING_HT) {

dict *ht = subject->ptr;

dictEntry *de = dictAddRaw(ht,value,NULL);

if (de) {

dictSetKey(ht,de,sdsdup(value));

dictSetVal(ht,de,NULL);

return 1;

}

}

else if (subject->encoding == OBJ_ENCODING_INTSET) {

if (isSdsRepresentableAsLongLong(value,&llval) == C_OK) {

uint8_t success = 0;

subject->ptr = intsetAdd(subject->ptr,llval,&success);

if (success) {

/* Convert to regular set when the intset contains

  • too many entries. */

if (intsetLen(subject->ptr) > server.set_max_intset_entries)

setTypeConvert(subject,OBJ_ENCODING_HT);

return 1;

}

}

else {

/* Failed to get integer from object, convert to regular set. */

setTypeConvert(subject,OBJ_ENCODING_HT);

/* The set was an intset and this value is not integer

  • encodable, so dictAdd should always work. */

serverAssert(dictAdd(subject->ptr,sdsdup(value),NULL) == DICT_OK);

return 1;

}

}

else {

serverPanic(“Unknown set encoding”);

}

return 0;

}

当执行sadd命令向集合中添加元素时,redis会校验待添加的元素是否可以解析为整数。如果解析失败,则会将集合存储结构转换为字典。


对象在不同情况下会采用不同的方式存储,那同时采用多种数据结构存储呢?也是会的。我们来看一下例子:

有序字典zset

typedef struct zset {

dict *dict;

zskiplist *zsl;

} zset;

最后

我还通过一些渠道整理了一些大厂真实面试主要有:蚂蚁金服、拼多多、阿里云、百度、唯品会、携程、丰巢科技、乐信、软通动力、OPPO、银盛支付、中国平安等初,中级,高级Java面试题集合,附带超详细答案,希望能帮助到大家。

新鲜出炉的蚂蚁金服面经,熬夜整理出来的答案,已有千人收藏

还有专门针对JVM、SPringBoot、SpringCloud、数据库、Linux、缓存、消息中间件、源码等相关面试题。

新鲜出炉的蚂蚁金服面经,熬夜整理出来的答案,已有千人收藏

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

Cloud、数据库、Linux、缓存、消息中间件、源码等相关面试题。**

[外链图片转存中…(img-feTEDtHC-1715048728638)]

本文已被CODING开源项目:【一线大厂Java面试题解析+核心总结学习笔记+最新讲解视频+实战项目源码】收录

需要这份系统化的资料的朋友,可以点击这里获取

  • 19
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值