Redis中的跳表是一种存储结构,它可以优化 Redis 的特定操作的性能,可用于存储有序的元素集合。事实上,Redis 中的跳表就是一个实现了元素排序的数据结构,它的实现基于一种叫做“跳表”的数据结构,该数据结构比标准的链表和数组数据结构更加灵活且快速,它能够根据关键字快速定位到元素,也能够快速获取有序列表中元素的范围,从而提高 Redis 的查找性能。

Redis 中的跳表是一个二叉树,包含一系列索引和“真实值”(元素值)的节点,每一个节点都有一个整数类型的索引和着一个指向“真实值”的指针。而“真实值”则是一个可以被外部存储的键值对,其中的 key 可以使用 O(1) 时间内获取到,value 则需要在 redis 中进行查找比较。

Redis 将每个跳表节点和“真实值”之间的关系建模为一种双向链表:每一个节点都指向一个“真实值”,而“真实值”又包含一个指向下一个节点的指针。这种链表构成了一种按顺序排列的数据结构,能够满足查找,范围查找,添加,删除等操作的需求。

下面是 Redis 跳表的部分源码,其中实现了 Redis 跳表的核心函数:

// 初始化跳表skiplist *skiplistCreate(void) {    struct skiplist *sl;    int j;    if ((sl = zmalloc(sizeof(*sl))) == NULL)        return NULL;    sl->level = 1;    sl->length = 0;    sl->header = skiplistNodeCreate(SKIPLIST_MAXLEVEL,0,NULL);    for (j = 0; j         sl->header->level[j].forward = NULL;        sl->header->level[j].span = 0;    }    sl->header->backward = NULL;    sl->tl = NULL;    return sl;}// 给定一个特定值,插入新节点int skiplistInsert(skiplist *sl, double score, sds ele) {    struct skiplistNode *update[SKIPLIST_MAXLEVEL], *x;    unsigned int level;    int i;    x = sl->header;    for (i = sl->level - 1; i >= 0; i--) {        while (x->level[i].forward &&             (x->level[i].forward->score             x = x->level[i].forward;        update[i] = x;    }    /* we assume the key is not already inside, since we allow duplicated     * scores, and the re-insertion of score and redis object should never     * happen since the caller of skiplistInsert() should test in the     * hash table first. */    level = skiplistRandomLevel();    if (level > sl->level) {        for (i = sl->level; i             update[i] = sl->header;            update[i]->level[i].span = sl->length;        }        sl->level = level;    }    x = skiplistNodeCreate(level,score,ele);    for (i = 0; i         x->level[i].forward = update[i]->level[i].forward;        update[i]->level[i].forward = x;        /* update span covered by update[i] as x is inserted here */        x->level[i].span = update[i]->level[i].span - (sl->length - update[i]->level[i].span);        update[i]->level[i].span = (sl->length - update[i]->level[i].span) + 1;    }    x->backward = (update[0] == sl->header) ? NULL : update[0];    if (x->level[0].forward)        x->level[0].forward->backward = x;    else        sl->tl = x;    sl->length++;    return 1;}

Redis 中的跳表支持 O(logn) 复杂度的查找,普通插入操作的复杂度也只需要 O(logn) ,在一定程度上可以帮助 Redis 提升性能。

香港服务器首选,2H2G首月10元开通。()提供简单好用,价格厚道的香港/美国云服务器和独立服务器。IDC+ISP+ICP资质。ARIN和APNIC会员。成熟技术团队15年行业经验。