揭秘Redis列表底层实现

Redis的列表数据结构是使用双向链表实现的,这使得在列表的两端进行插入和删除操作都可以在常数时间内完成。

在Redis内部,列表的每个节点使用一个listNode结构表示:




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

而整个列表由一个list结构体来维护:




typedef struct list {
    listNode *head;
    listNode *tail;
    void (*free)(void *ptr);
    unsigned long len;
} list;

其中:

  • head 指向列表的头节点。
  • tail 指向列表的尾节点。
  • free 是一个函数指针,用于释放列表节点的值所占用的内存。
  • len 记录了列表的长度。

以下是一个简单的Redis列表结构的实现示例:




#include <stdio.h>
#include <stdlib.h>
 
typedef struct listNode {
    struct listNode *prev;
    struct listNode *next;
    void *value;
} listNode;
 
typedef struct list {
    listNode *head;
    listNode *tail;
    void (*free)(void *ptr);
    unsigned long len;
} list;
 
list *listCreate() {
    list *l = malloc(sizeof(list));
    l->head = NULL;
    l->tail = NULL;
    l->len = 0;
    l->free = NULL;
    return l;
}
 
listNode *listNodeCreate(void *value) {
    listNode *node = malloc(sizeof(listNode));
    node->prev = NULL;
    node->next = NULL;
    node->value = value;
    return node;
}
 
int listAddNodeHead(list *l, void *value) {
    listNode *node = listNodeCreate(value);
    if (l->len == 0) {
        l->head = node;
        l->tail = node;
    } else {
        node->next = l->head;
        l->head->prev = node;
        l->head = node;
    }
    l->len++;
    return 1;
}
 
int listAddNodeTail(list *l, void *value) {
    listNode *node = listNodeCreate(value);
    if (l->len == 0) {
        l->head = node;
        l->tail = node;
    } else {
        node->prev = l->tail;
        l->tail->next = node;
        l->tail = node;
    }
    l->len++;
    return 1;
}
 
void listFree(list *l) {
    listNode *current = l->head;
    listNode *next = NULL;
    while(current != NULL) {
        next = current->next;
        if (l->free) l->free(current->value);
        free(current);
        current = next;
    }
    free(l);
}
 
int main() {
    list *myList = listCreate();
    listAddNodeHead(myList, "Hello");
    listAddNodeTail(myList, "World");
    // 现在myList包含两个节点,分别保存"Hello"和"World"
 
    // 清理资源
    listFree(myList);
    return 0;
}

这个示例展示了如何创建一个列表,如何添加节点到列表的头部和尾部,以及如何释放整个列表所占用的内存。这里的代码仅为示例,并未包含完整的错误检查和处理。

最后修改于:2024年09月06日 10:07

评论已关闭

推荐阅读

DDPG 模型解析,附Pytorch完整代码
2024年11月24日
DQN 模型解析,附Pytorch完整代码
2024年11月24日
AIGC实战——Transformer模型
2024年12月01日
Socket TCP 和 UDP 编程基础(Python)
2024年11月30日
python , tcp , udp
如何使用 ChatGPT 进行学术润色?你需要这些指令
2024年12月01日
AI
最新 Python 调用 OpenAi 详细教程实现问答、图像合成、图像理解、语音合成、语音识别(详细教程)
2024年11月24日
ChatGPT 和 DALL·E 2 配合生成故事绘本
2024年12月01日
omegaconf,一个超强的 Python 库!
2024年11月24日
【视觉AIGC识别】误差特征、人脸伪造检测、其他类型假图检测
2024年12月01日
[超级详细]如何在深度学习训练模型过程中使用 GPU 加速
2024年11月29日
Python 物理引擎pymunk最完整教程
2024年11月27日
MediaPipe 人体姿态与手指关键点检测教程
2024年11月27日
深入了解 Taipy:Python 打造 Web 应用的全面教程
2024年11月26日
基于Transformer的时间序列预测模型
2024年11月25日
Python在金融大数据分析中的AI应用(股价分析、量化交易)实战
2024年11月25日
AIGC Gradio系列学习教程之Components
2024年12月01日
Python3 `asyncio` — 异步 I/O,事件循环和并发工具
2024年11月30日
llama-factory SFT系列教程:大模型在自定义数据集 LoRA 训练与部署
2024年12月01日
Python 多线程和多进程用法
2024年11月24日
Python socket详解,全网最全教程
2024年11月27日