首页 > 数据库 >Redis 数据结构 - 链表

Redis 数据结构 - 链表

时间:2023-07-12 12:00:12浏览次数:41  
标签:node NULL Redis list next 链表 数据结构 节点

链表 - List 的底层实现

链表提供了高效的节点重排能力,可以通过顺序访问的方式访问节点,并且支持增加删除节点调整长度。

由于 C 语言原生并不支持链表,redis 的链表是自己实现的。

List 的底层实现就是一个双向链表,支持从链表的两端进行pushpop操作,时间复杂度是O(1)。同时支持在任意位置进行存取操作,但是需要进行整体的遍历,从而时间复杂度为O(n)。List 的应用场景很多,通常被用作队列使用。同时,微博的关注列表,粉丝列表,消息队列等功能都可以用 redis 的 list 结构来实现。可以利用 lrange 命令,做基于 redis 的分页功能。

除了 List 外,发布订阅、慢查询、监视器等功能也都用到了链表,Redis 本身服务器还使用链表来保存多个客户端的状态信息,以及使用链表来构建客户端输出缓冲区(output buffer)。

链表和链表节点的实现

/* Node, List, and Iterator are the only data structures used currently. */
/*
 * 双端链表节点
 */
typedef struct listNode {
    // 前置节点
    struct listNode *prev;
    // 后置节点
    struct listNode *next;
    // 节点的值
    void *value;
} listNode;

每个节点可以知晓自己的前置节点和后置节点,多个 listNode 就可以组成一个双向链表,但是 redis 额外提供了一个 list 用来持有链表,提供了额外的操作

/*
 * 双端链表结构
 */
typedef struct list {
    // 表头节点
    listNode *head;
    // 表尾节点
    listNode *tail;
    // 节点值复制函数,用来赋值链表节点所保存的值
    void *(*dup)(void *ptr);
    // 节点值释放函数,用来释放链表节点所保存的值
    void (*free)(void *ptr);
    // 节点值对比函数,用来比对链表节点所保存的值和另一个输入值是否相等
    int (*match)(void *ptr, void *key);
    // 链表所包含的节点数量
    unsigned long len;
} list;

所以一个链表结构由 一个 list 结构和若干个 listNode 结构组成:

由结构可知,redis 的链表实现具有一下特性:

  1. 双端
    • 链表自带头指针 prev 和尾指针 next,获取某个节点的前置节点和后置节点的时间复杂度都是 O(1)。
  2. 无环
    • 表头节点的 prev 指针和表尾节点 next 指针都指向 null,对链表的访问以 null 为终点。
  3. 带表头指针和表尾指针
    • 通过 list 结构的 head 指针和 tail 指针,程序获取链表的头节点和尾节点的时间复杂度为 O(1)。
  4. 多态
    • 链表节点使用 void* 指针来保存节点值,并且可以通过 list 结构的 dup、free、match 三个属性为节点值实现特定的类型函数,所以链表可以用来保存不同的值。list 本身可以视为一个抽象。

链表和链表节点的 API

创建

/* Create a new list. The created list can be freed with
 * listRelease(), but private value of every node need to be freed
 * by the user before to call listRelease(), or by setting a free method using
 * listSetFreeMethod.
 *
 * On error, NULL is returned. Otherwise the pointer to the new list. */
list *listCreate(void)
{
    struct list *list;
		// 分配内存
    if ((list = zmalloc(sizeof(*list))) == NULL)
        return NULL;
  	// 初始化属性
    list->head = list->tail = NULL;
    list->len = 0;
    list->dup = NULL;
    list->free = NULL;
    list->match = NULL;
    return list;
}

释放

/* Remove all the elements from the list without destroying the list itself. */
void listEmpty(list *list)
{
    unsigned long len;
    listNode *current, *next;
		// 从头指针开始遍历
    current = list->head;
    len = list->len;
    while(len--) {
        next = current->next;
      	// 如果存在释放函数,那么执行它,释放节点值
        if (list->free) list->free(current->value);
        // 释放节点自身结构
      	zfree(current);
        current = next;
    }
  	// 头、尾节点置为空
    list->head = list->tail = NULL;
    list->len = 0;
}

/* Free the whole list.
 *
 * This function can't fail. */
void listRelease(list *list)
{
  	// 删除所有链表元素
    listEmpty(list);
  	// 释放链表结构
    zfree(list);
}

添加一个节点

添加到头位置
/* Add a new node to the list, to head, containing the specified 'value'
 * pointer as value.
 *
 * On error, NULL is returned and no operation is performed (i.e. the
 * list remains unaltered).
 * On success the 'list' pointer you pass to the function is returned. */
list *listAddNodeHead(list *list, void *value)
{
    listNode *node;
		// 分配内存错误返回 null
    if ((node = zmalloc(sizeof(*node))) == NULL)
        return NULL;
  	// 新的节点记录节点值
    node->value = value;
  	// 
    listLinkNodeHead(list, node);
    return list;
}

/*
 * Add a node that has already been allocated to the head of list
 */
void listLinkNodeHead(list* list, listNode *node) {
    if (list->len == 0) {
      	// 链表是空的,追加到链表头
        list->head = list->tail = node;
        node->prev = node->next = NULL;
    } else {
      	// 链表是非空的,追加到链表头
        node->prev = NULL;
        node->next = list->head;
        list->head->prev = node;
        list->head = node;
    }
    list->len++;
}
添加到尾位置
/* Add a new node to the list, to tail, containing the specified 'value'
 * pointer as value.
 *
 * On error, NULL is returned and no operation is performed (i.e. the
 * list remains unaltered).
 * On success the 'list' pointer you pass to the function is returned. */
list *listAddNodeTail(list *list, void *value)
{
    listNode *node;

    if ((node = zmalloc(sizeof(*node))) == NULL)
        return NULL;
    node->value = value;
    listLinkNodeTail(list, node);
    return list;
}

/*
 * Add a node that has already been allocated to the tail of list
 */
void listLinkNodeTail(list *list, listNode *node) {
    if (list->len == 0) {
        list->head = list->tail = node;
        node->prev = node->next = NULL;
    } else {
        node->prev = list->tail;
        node->next = NULL;
        list->tail->next = node;
        list->tail = node;
    }
    list->len++;
}

指定节点(前\后)

/*
 * 创建一个包含值 value 的新节点,并将它插入到 old_node 的之前或之后
 *
 * 如果 after 为 0 ,将新节点插入到 old_node 之前。
 * 如果 after 为 1 ,将新节点插入到 old_node 之后。
 *
 * T = O(1)
 */
list *listInsertNode(list *list, listNode *old_node, void *value, int after) {
    listNode *node;

    // 创建新节点
    if ((node = zmalloc(sizeof(*node))) == NULL)
        return NULL;

    // 保存值
    node->value = value;

    // 将新节点添加到给定节点之后
    if (after) {
        node->prev = old_node;
        node->next = old_node->next;
        // 给定节点是原表尾节点
        if (list->tail == old_node) {
            list->tail = node;
        }
    // 将新节点添加到给定节点之前
    } else {
        node->next = old_node;
        node->prev = old_node->prev;
        // 给定节点是原表头节点
        if (list->head == old_node) {
            list->head = node;
        }
    }

    // 更新新节点的前置指针
    if (node->prev != NULL) {
        node->prev->next = node;
    }
    // 更新新节点的后置指针
    if (node->next != NULL) {
        node->next->prev = node;
    }

    // 更新链表节点数
    list->len++;

    return list;
}

释放一个节点

/* Remove the specified node from the specified list.
 * The node is freed. If free callback is provided the value is freed as well.
 *
 * This function can't fail. */
void listDelNode(list *list, listNode *node)
{
  	// 修改链表
    listUnlinkNode(list, node);
  	// 如果提供了值释放函数,那么释放节点值
    if (list->free) list->free(node->value);
  	// 释放节点结构
    zfree(node);
}

/*
 * Remove the specified node from the list without freeing it.
 */
void listUnlinkNode(list *list, listNode *node) {
  	// 调整前置节点
    if (node->prev)
      	// 非头节点
        node->prev->next = node->next;
    else
      	// 头节点
        list->head = node->next;
  	// 调整后置节点 
    if (node->next)
      	// 非尾节点
        node->next->prev = node->prev;
    else
      	// 尾节点
        list->tail = node->prev;
		// 置空指针
    node->next = NULL;
    node->prev = NULL;
		// 链表长度减少
    list->len--;
}

迭代

/* Returns a list iterator 'iter'. After the initialization every
 * call to listNext() will return the next element of the list.
 *
 * This function can't fail. */
listIter *listGetIterator(list *list, int direction)
{
    listIter *iter;

    if ((iter = zmalloc(sizeof(*iter))) == NULL) return NULL;
    if (direction == AL_START_HEAD)
        iter->next = list->head;
    else
        iter->next = list->tail;
    iter->direction = direction;
    return iter;
}

/* Release the iterator memory */
void listReleaseIterator(listIter *iter) {
    zfree(iter);
}

/* Create an iterator in the list private iterator structure */
void listRewind(list *list, listIter *li) {
    li->next = list->head;
    li->direction = AL_START_HEAD;
}

void listRewindTail(list *list, listIter *li) {
    li->next = list->tail;
    li->direction = AL_START_TAIL;
}

查找

/* Search the list for a node matching a given key.
 * The match is performed using the 'match' method
 * set with listSetMatchMethod(). If no 'match' method
 * is set, the 'value' pointer of every node is directly
 * compared with the 'key' pointer.
 *
 * On success the first matching node pointer is returned
 * (search starts from head). If no matching node exists
 * NULL is returned. */
listNode *listSearchKey(list *list, void *key)
{
    listIter iter;
    listNode *node;

    listRewind(list, &iter);
    while((node = listNext(&iter)) != NULL) {
        if (list->match) {
            if (list->match(node->value, key)) {
                return node;
            }
        } else {
            if (key == node->value) {
                return node;
            }
        }
    }
    return NULL;
}

查找链表 list 中值和 key 匹配的节点。对比操作由链表的 match 函数负责进行,如果没有设置 match 函数,那么直接通过对比值的指针来决定是否匹配。如果匹配成功,那么第一个匹配的节点会被返回。如果没有匹配任何节点,那么返回 NULL 。时间复杂度 T = O(N)。

/* Return the element at the specified zero-based index
 * where 0 is the head, 1 is the element next to head
 * and so on. Negative integers are used in order to count
 * from the tail, -1 is the last element, -2 the penultimate
 * and so on. If the index is out of range NULL is returned. */
listNode *listIndex(list *list, long index) {
    listNode *n;
		
  	// 如果索引为负数,从表尾开始查找
    if (index < 0) {
        index = (-index)-1;
        n = list->tail;
        while(index-- && n) n = n->prev;
    } else {
      	// 如果索引为正数,从表头开始查找
        n = list->head;
        while(index-- && n) n = n->next;
    }
    return n;
}

拼接

/* Add all the elements of the list 'o' at the end of the
 * list 'l'. The list 'other' remains empty but otherwise valid. */
// 默认将 list o 的所有元素追加到 list l 的尾
void listJoin(list *l, list *o) {
    if (o->len == 0) return;

    o->head->prev = l->tail;

    if (l->tail)
        l->tail->next = o->head;
    else
        l->head = o->head;

    l->tail = o->tail;
    l->len += o->len;

    /* Setup other as an empty list. */
    o->head = o->tail = NULL;
    o->len = 0;
}

总结

redis 的链表,是一个双端、无环、支持多态的用一个 list 结构表示的链表。

标签:node,NULL,Redis,list,next,链表,数据结构,节点
From: https://www.cnblogs.com/radish40/p/17547156.html

相关文章

  • 如何实现redis面试的具体操作步骤
    实现Redis面试指南简介在面试过程中,Redis是一个常见的面试题目,因此了解如何实现“Redis面试”是非常重要的。本文将指导你完成这个任务。流程概述下面是实现“Redis面试”任务的流程概述:步骤描述步骤1配置Redis环境并安装Redis依赖步骤2创建一个Redis......
  • 如何实现安装redis的具体操作步骤
    如何安装Redis简介Redis是一个开源的高性能键值数据库,广泛用于缓存、消息队列、排行榜等场景。本文将介绍如何在Linux环境下安装Redis,并提供相应的代码示例。安装步骤步骤描述1.安装依赖安装Redis所需的依赖库2.下载Redis从Redis官网下载最新版本的Redis源代码......
  • Redis 客户端中查不到数据的解决方法
    问题:Java代码中能获取到redis数据,但是在服务器中使用redis-cli登录redis客户端后,使用get等命令获取不到数据。原因:没有选择数据库,查看java代码的配置后,发现使用的是1号数据库,但是命令行登录进去redis后默认是0号数据库,因此就查不到数据。解决:使用命令select1,选择1......
  • Redis 命令行中报错 (error) NOAUTH Authentication required
    本文来源:redis客户端连接错误NOAUTHAuthenticationrequired_Redis_脚本之家redis客户端连接成功,但是操作报异常——(error)NOAUTHAuthenticationrequired错误的含义是说你没有认证,说明没有使用密码连接查看密码:进入redis的安装目录,查看redis.config文件,viredis.conf......
  • C#连接Redis - Redis教程 (yiibai.com) (转)
    C#连接Redis-Redis教程(yiibai.com)classProgram{staticvoidMain(string[]args){//在Redis中存储常用的5种数据类型:String,Hash,List,SetSortedsetvarclient=newRedisClient("127.0.0.1",6379);//A......
  • docker 安装redis 6.0.8哨兵集群(一主两从三哨兵)
    准备三台主机并且安装了docker192.168.31.132192.168.31.134192.168.31.144linux版redis6.0.8下载下载地址:https://download.redis.io/releases/干啥用:拷贝出redis.conf文件,在此文件里配置主从关系,最好不要使用不同版本的配置文件,防止出现配置文件的参数不兼容问题安......
  • Redis CRUD Client
    #-*-coding:utf-8-*-importredisfromconfig.redis_configimportCACHE_REDIS_CONF#CACHE_REDIS_CONF={#"host":"x.x.x.x",#"port":????,#"password":??????,#"db":??,#}redis......
  • 数据结构与算法 #18 下跳棋,极富想象力的同向双指针模拟
    ⭐️本文已收录到AndroidFamily,技术和职场问题,请关注公众号[彭旭锐]和[BaguTreePro]知识星球提问。学习数据结构与算法的关键在于掌握问题背后的算法思维框架,你的思考越抽象,它能覆盖的问题域就越广,理解难度也更复杂。在这个专栏里,小彭将基于Java/Kotlin语言,为你分享常......
  • 反转链表
    给定一个单链表的头结点pHead(该头节点是有值的,比如在下图,它的val是1),长度为n,反转该链表后,返回新链表的表头。 如当输入链表{1,2,3}时,经反转后,原链表变为{3,2,1},所以对应的输出为{3,2,1}。以上转换过程如下图所示:......
  • 数据结构(第七章)
    数据结构(第七章)基本概念查找:在数据集合中寻找满足某种条件的数据元素的过程称为查找。查找表:用于查找的数据集合称为查找表平均查找长度:在查找过程中,一次查找的长度是指需要比较的关键字次数,而平均查找长度则是所有查找过程中进行关键字的比较次数的平均值。线性表查找一......