`

Redis代码阅读2--Redis数据结构之链表

 
阅读更多

     Redis 是一个开源的高性能key-value数据库,其很大程度上弥补了memeched这类key-value存储的不足(除了支持String外,还支持Hash,Set,sorted set, List),在部分场合对关系型数据库也起到了很好的补充作用。因为Redis的代码量并不多,因为我逐步阅读了其源代码,以期能对其有深入的理解。首先介绍Redis支持的各种数据结构。

     Redis中的链表是双向链表,这一点可以由 adlist.h 中代码得出:

typedef struct listNode {
    struct listNode *prev;
    struct listNode *next;
    void *value;
} listNode;

typedef struct listIter {
    listNode *next;
    int direction;
} listIter;

typedef struct list {
    listNode *head;
    listNode *tail;
    void *(*dup)(void *ptr);
    void (*free)(void *ptr);
    int (*match)(void *ptr, void *key);
    unsigned int len;
} list;

 struct list通过head 和tail这两个指针维护了链表的头、尾节点。ListNode通过prev和next使得链表成为一个双向链表。Struct List里面的dup、free和match这三个函数指针用于配置复制、释放和匹配三个功能。比如listSetMatchMethod就是用来设置match方法的。

  Redis里的List这一数据结构和我们常说的数据结构里面的链表很相似,同样有在头、尾节点增加Node,删除Node,查找Node及Index等功能。

  Redis里面操作list的命令有RPUSH,LPUSH,LSET,LLEN等。Redis中负责实现这些命令的API就在t_list.c。下面就以RPUSH为例,来说明当客户端执行RPUSH命令时,Redis是如何通过list 的API去执行list的listAddNodeTail方法的。

   在Redis.c里面,有一个Struct里面叫readonlyCommandTable。这个struct将Redis的各种命令和Redis里面各种命令对应的函数关联起来。比如:{"rpush",rpushCommand,-3,REDIS_CMD_DENYOOM,NULL,1,1,1}。

   Redis Server接收各个Client传过来的各种命令是通过networking.c里面的代码来完成的。具体如何解析和执行各种Command以后会单独拎出来讲。这里主要简单介绍下其流程。

 processInputBuffer-->processCommand-->lookupCommand.这里的lookupCommand就是根据Command的名字从readonlyCommandTable找出该Command对应的方法。比如,Client执行RPUSH命令,则这里的lookupCommand找到的是rpushCommand。

 

void rpushCommand(redisClient *c) {
    pushGenericCommand(c,REDIS_TAIL);
}

void pushGenericCommand(redisClient *c, int where) {
    int j, addlen = 0, pushed = 0;
    robj *lobj = lookupKeyWrite(c->db,c->argv[1]);
    int may_have_waiting_clients = (lobj == NULL);

    if (lobj && lobj->type != REDIS_LIST) {
        addReply(c,shared.wrongtypeerr);
        return;
    }

    for (j = 2; j < c->argc; j++) {
        c->argv[j] = tryObjectEncoding(c->argv[j]);
        if (may_have_waiting_clients) {
            if (handleClientsWaitingListPush(c,c->argv[1],c->argv[j])) {
                addlen++;
                continue;
            } else {
                may_have_waiting_clients = 0;
            }
        }
        if (!lobj) {
            lobj = createZiplistObject();
            dbAdd(c->db,c->argv[1],lobj);
        }
        listTypePush(lobj,c->argv[j],where);
        pushed++;
    }
    addReplyLongLong(c,addlen + (lobj ? listTypeLength(lobj) : 0));
    if (pushed) signalModifiedKey(c->db,c->argv[1]);
    server.dirty += pushed;
}

void listTypePush(robj *subject, robj *value, int where) {
    /* Check if we need to convert the ziplist */
    listTypeTryConversion(subject,value);
    if (subject->encoding == REDIS_ENCODING_ZIPLIST &&
        ziplistLen(subject->ptr) >= server.list_max_ziplist_entries)
            listTypeConvert(subject,REDIS_ENCODING_LINKEDLIST);

    if (subject->encoding == REDIS_ENCODING_ZIPLIST) {
        int pos = (where == REDIS_HEAD) ? ZIPLIST_HEAD : ZIPLIST_TAIL;
        value = getDecodedObject(value);
        subject->ptr = ziplistPush(subject->ptr,value->ptr,sdslen(value->ptr),pos);
        decrRefCount(value);
    } else if (subject->encoding == REDIS_ENCODING_LINKEDLIST) {
        if (where == REDIS_HEAD) {
            listAddNodeHead(subject->ptr,value);
        } else {
            listAddNodeTail(subject->ptr,value);
        }
        incrRefCount(value);
    } else {
        redisPanic("Unknown list encoding");
    }
}

     在t_list.c里面,rpushCommand-->pushGenericCommand-->listTypePush。listTypePush里面通过方向变量where来判断是在head还是在tail插入新的Node。

 

 

 

 

 

0
0
分享到:
评论

相关推荐

    Redis-x64-5.0.14.1.msi

    1. **Redis 简介**:Redis 是 Remote Dictionary Server 的缩写,由 Salvatore Sanfilippo 创建,最初设计为内存数据结构存储系统,支持多种数据结构如字符串、哈希表、列表、集合和有序集合。它通过持久化机制确保...

    Redis-x64-3.0.504.zip

    它是一款开源、免费的NoSQL数据库,支持多种数据结构如字符串、哈希表、列表、集合和有序集合。 2. **数据类型**: - 字符串:基本数据类型,可存储任意长度的字符串。 - 哈希:用于存储键值对,适合表示对象。 ...

    Redis-x64-5.0.9.msi

    redis是一个key-value存储系统。和Memcached类似,它支持存储的value类型相对更多,...区别的是redis会周期性的把更新的数据写入磁盘或者把修改操作写入追加的记录文件,并且在此基础上实现了master-slave(主从)同步。

    Redis-x64-5.0.14.1.zip

    Redis是一个基于内存的数据结构存储系统,它支持多种数据类型,如字符串、哈希、列表、集合和有序集合。这些数据结构使得Redis能够处理各种复杂的应用场景,例如计数、发布订阅、限速等。 2. **Redis的数据类型** ...

    01-VIP-Redis核心数据结构与核心原理1

    Redis是一款高性能的键值存储系统,它以其丰富的数据结构、高效的数据操作以及强大的持久化机制在现代互联网架构中扮演着重要角色。本篇文章主要探讨Redis的核心数据结构和核心原理,以及如何利用IO多路复用技术处理...

    Redis-x64-5.0.10

    windows下redis安装包。 redis是一个key-value存储系统。它支持存储的value类型很多,包括string(字符串)、list(链表)、set(集合)、zset(sorted set --有序集合)和hash(哈希类型)。这些数据类型都支持push/pop、...

    Redis-x64-4.0.14

    redis安装版 直接可以运行在windows端。。redis是一个key-value...和Memcached类似,它支持存储的value类型相对更多,包括string(字符串)、list(链表)、set(集合)、zset(sorted set --有序集合)和hash(哈希类型)。

    Redis-x64-3.2.100.zip

    和Memcached类似,它支持存储的value类型相对更多,包括string(字符串)、list(链表)、set(集合)、zset(sorted set --有序集合)和hash(哈希类型)。这些数据类型都支持push/pop、add/remove及取交集并集和差集及更...

    redis-x64-3.2.1

    和Memcached类似,它支持存储的value类型相对更多,包括string(字符串)、list(链表)、set(集合)、zset(sorted set --有序集合)和hash(哈希类型)。这些数据类型都支持push/pop、add/remove及取交集并集和差集及更...

    Redis-x64-3.2.100.rar

    同时,Redis支持多种数据结构,如字符串、哈希、列表、集合、有序集合等,使得其在处理复杂的数据操作上表现出色。 二、Redis在Windows上的安装 1. 解压"Redis-x64-3.2.100.rar",得到Redis的Windows版本。 2. 配置...

    Redis实战-food-social-contact-parent.zip

    1. Redis数据结构: - String:基础数据类型,用于存储字符串、数字等简单数据。 - Hash:用于存储键值对,适用于用户资料存储,如用户ID与用户名、联系方式等。 - Set:无序不重复元素集合,可以用来存储关注...

    Redis-x64-2.8.2104.zip (windows下不需安装解压版)

    和Memcached类似,它支持存储的value类型相对更多,包括string(字符串)、list(链表)、set(集合)、zset(sorted set --有序集合)和hash(哈希类型)。这些数据类型都支持push/pop、add/remove及取交集并集和差集及更...

    Redis-x64-3.2.100 压缩版 windows版本

    - Redis是Remote Dictionary Server(远程字典服务器)的缩写,它以键值对的形式存储数据,支持多种数据结构,如字符串、哈希、列表、集合和有序集合。 - Redis以其内存存储特性而闻名,数据读取速度非常快,因为...

    Redis-for-windows-x64-2.8.2400

    和Memcached类似,它支持存储的value类型相对更多,包括string(字符串)、list(链表)、set(集合)、zset(sorted set --有序集合)和hash(哈希类型)。这些数据类型都支持push/pop、add/remove及取交集并集和差集及更...

    Redis-x64-3.2.100 及redis安装使用文档

    Redis以其高效的性能、丰富的数据结构和简单易用的特性,在IT行业中广泛应用。以下是对Redis-x64-3.2.100版本的安装和使用进行的详细解释。 首先,我们来看Redis的安装过程。对于Windows平台,Redis-x64-3.2.100是...

    Redis-x64-3.0.503.rar

    Redis 是一个开源的、基于键值对的数据存储系统,它被广泛用于缓存、数据库以及消息中间件等场景。此压缩包 "Redis-x64-3.0.503.rar" 提供的是适用于 Windows 操作系统的 Redis 3.0.503 版本。在Windows环境下使用...

Global site tag (gtag.js) - Google Analytics