Redis深度历险-集合

2021-08-08  本文已影响0人  突击手平头哥

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");
    }
}
上一篇 下一篇

猜你喜欢

热点阅读