Redis深度历险-集合
Redis中的set是一种无序集合,如果存储的全部都是数字则内部使用的是intset存储,否则使用的是hashtab存储
字符串集合
int setTypeAdd(robj *subject, sds value) {
long long llval;
if (subject->encoding == OBJ_ENCODING_HT) {
dict *ht = subject->ptr;
//使用的是字典存储,只不过value是NULL
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) {
..........
} else {
serverPanic("Unknown set encoding");
}
return 0;
}
在编码是OBJ_ENCODING_HT
时直接使用字典来存储,只不过所有的value是NULL
数字集合
int setTypeAdd(robj *subject, sds value) {
long long llval;
if (subject->encoding == OBJ_ENCODING_HT) {
.........
} else if (subject->encoding == OBJ_ENCODING_INTSET) {
//判断能够转为long long类型的证书
if (isSdsRepresentableAsLongLong(value,&llval) == C_OK) {
uint8_t success = 0;
//插入到intset数据结构中
subject->ptr = intsetAdd(subject->ptr,llval,&success);
if (success) {
//如果intset中存储的数据超过set_max_intset_entries则转为字典存储,默认512个
if (intsetLen(subject->ptr) > server.set_max_intset_entries)
setTypeConvert(subject,OBJ_ENCODING_HT);
return 1;
}
} else {
//不能转换成证书,则转为存储到字典中
setTypeConvert(subject,OBJ_ENCODING_HT);
serverAssert(dictAdd(subject->ptr,sdsdup(value),NULL) == DICT_OK);
return 1;
}
} else {
serverPanic("Unknown set encoding");
}
return 0;
}
编码类型转换
在插入字符串或者数据数量超过512个(配置文件中设置)时转为为
OBJ_ENCODING_HT
void setTypeConvert(robj *setobj, int enc) {
setTypeIterator *si;
serverAssertWithInfo(NULL,setobj,setobj->type == OBJ_SET &&
setobj->encoding == OBJ_ENCODING_INTSET);
//只支持转换为字典类型
if (enc == OBJ_ENCODING_HT) {
int64_t intele;
dict *d = dictCreate(&setDictType,NULL);
sds element;
//按集合数据类型,初始化字典空间
dictExpand(d,intsetLen(setobj->ptr));
//将intset中的每一项数据逐个加入到集合中
si = setTypeInitIterator(setobj);
while (setTypeNext(si,&element,&intele) != -1) {
element = sdsfromlonglong(intele);
serverAssert(dictAdd(d,element,NULL) == DICT_OK);
}
setTypeReleaseIterator(si);
//释放原油的空间
setobj->encoding = OBJ_ENCODING_HT;
zfree(setobj->ptr);
setobj->ptr = d;
} else {
serverPanic("Unsupported set conversion");
}
}
网友评论