0

0

C语言中怎样实现LRU缓存 C语言哈希表与双向链表结合应用

穿越時空

穿越時空

发布时间:2025-07-01 11:15:02

|

982人浏览过

|

来源于php中文网

原创

c语言实现lru缓存的核心在于结合哈希表与双向链表。1. 哈希表用于快速查找,时间复杂度为o(1);2. 双向链表维护访问顺序,最近使用项置于头部,最久未用项置于尾部;3. 缓存项结构包含key、value及前后指针;4. 初始化时分配内存并初始化哈希表和互斥锁;5. 获取缓存时命中则移动至链表头部;6. 设置缓存时若存在则更新并移动,否则新建节点插入头部并可能淘汰尾部节点;7. 使用链地址法处理哈希冲突,头插法插入节点;8. 通过添加pthread互斥锁解决线程安全问题,在操作缓存前加锁,操作后解锁;9. 哈希函数选择需考虑均匀性、高效性和简单性,如murmurhash、fnv-1a或djb2等,以提升性能。

C语言中怎样实现LRU缓存 C语言哈希表与双向链表结合应用

C语言实现LRU缓存,核心在于结合哈希表快速查找和双向链表维护访问顺序。哈希表用于O(1)时间复杂度查找缓存项,双向链表则用于记录缓存项的使用情况,最近使用的放在链表头部,最久未使用的放在尾部。

C语言中怎样实现LRU缓存 C语言哈希表与双向链表结合应用

解决方案

C语言中怎样实现LRU缓存 C语言哈希表与双向链表结合应用
  1. 数据结构定义:

    立即学习C语言免费学习笔记(深入)”;

    typedef struct CacheNode {
        char *key;
        void *value;
        struct CacheNode *prev;
        struct CacheNode *next;
    } CacheNode;
    
    typedef struct LRUCache {
        size_t capacity;
        size_t size;
        CacheNode *head;
        CacheNode *tail;
        // 哈希表,存储 key -> CacheNode* 的映射
        CacheNode **table;
    } LRUCache;
  2. 初始化LRU缓存:

    C语言中怎样实现LRU缓存 C语言哈希表与双向链表结合应用
    LRUCache* lruCacheCreate(size_t capacity) {
        LRUCache* cache = (LRUCache*)malloc(sizeof(LRUCache));
        cache->capacity = capacity;
        cache->size = 0;
        cache->head = NULL;
        cache->tail = NULL;
        cache->table = (CacheNode**)calloc(capacity, sizeof(CacheNode*)); // 哈希表初始化
        return cache;
    }
  3. 哈希函数(简单示例):

    unsigned long hash(const char *key) {
        unsigned long hash = 5381;
        int c;
    
        while ((c = *key++))
            hash = ((hash << 5) + hash) + c; /* hash * 33 + c */
    
        return hash;
    }
  4. 获取缓存项:

    void* lruCacheGet(LRUCache* cache, const char *key) {
        unsigned long index = hash(key) % cache->capacity;
        CacheNode* node = cache->table[index];
    
        // 查找哈希表中是否存在该key
        while (node != NULL) {
            if (strcmp(node->key, key) == 0) {
                // 命中缓存,将节点移动到链表头部
                // 1. 从链表中移除
                if (node->prev != NULL) {
                    node->prev->next = node->next;
                } else {
                    cache->head = node->next; // 如果是头节点,更新头节点
                }
    
                if (node->next != NULL) {
                    node->next->prev = node->prev;
                } else {
                    cache->tail = node->prev; // 如果是尾节点,更新尾节点
                }
    
                // 2. 将节点添加到链表头部
                node->prev = NULL;
                node->next = cache->head;
                if (cache->head != NULL) {
                    cache->head->prev = node;
                }
                cache->head = node;
    
                if (cache->tail == NULL) {
                    cache->tail = node; // 如果是第一个节点,同时更新尾节点
                }
    
                return node->value;
            }
            node = node->next; // 哈希冲突,链表解决
        }
    
        return NULL; // 未找到
    }
  5. 设置缓存项:

    void lruCachePut(LRUCache* cache, const char *key, void *value) {
        unsigned long index = hash(key) % cache->capacity;
        CacheNode* existingNode = cache->table[index];
    
        // 检查key是否已存在
        while (existingNode != NULL) {
            if (strcmp(existingNode->key, key) == 0) {
                // Key已存在,更新value并移动到链表头部
                existingNode->value = value;
                // 移动到头部 (同get操作)
                if (existingNode->prev != NULL) {
                    existingNode->prev->next = existingNode->next;
                } else {
                    cache->head = existingNode->next;
                }
    
                if (existingNode->next != NULL) {
                    existingNode->next->prev = existingNode->prev;
                } else {
                    cache->tail = existingNode->prev;
                }
    
                existingNode->prev = NULL;
                existingNode->next = cache->head;
                if (cache->head != NULL) {
                    cache->head->prev = existingNode;
                }
                cache->head = existingNode;
    
                if (cache->tail == NULL) {
                    cache->tail = existingNode;
                }
                return;
            }
            existingNode = existingNode->next; // 处理哈希冲突
        }
    
        // Key不存在,创建新节点
        CacheNode* newNode = (CacheNode*)malloc(sizeof(CacheNode));
        newNode->key = strdup(key); // 复制key,避免外部修改
        newNode->value = value;
        newNode->prev = NULL;
        newNode->next = cache->head;
    
        if (cache->head != NULL) {
            cache->head->prev = newNode;
        }
        cache->head = newNode;
    
        if (cache->tail == NULL) {
            cache->tail = newNode;
        }
    
        // 添加到哈希表
        newNode->next = cache->table[index]; // 头插法解决哈希冲突
        cache->table[index] = newNode;
    
        cache->size++;
    
        // 如果超出容量,移除链表尾部节点
        if (cache->size > cache->capacity) {
            CacheNode* tailNode = cache->tail;
            if (tailNode != NULL) {
                // 1. 从链表中移除
                cache->tail = tailNode->prev;
                if (cache->tail != NULL) {
                    cache->tail->next = NULL;
                } else {
                    cache->head = NULL; // 链表为空
                }
    
                // 2. 从哈希表中移除(需要遍历哈希表对应链表)
                unsigned long tailIndex = hash(tailNode->key) % cache->capacity;
                CacheNode* current = cache->table[tailIndex];
                CacheNode* prev = NULL;
                while (current != NULL) {
                    if (current == tailNode) {
                        if (prev == NULL) {
                            cache->table[tailIndex] = current->next; // 移除头节点
                        } else {
                            prev->next = current->next;
                        }
                        break;
                    }
                    prev = current;
                    current = current->next;
                }
    
                free(tailNode->key);
                free(tailNode);
                cache->size--;
            }
        }
    }
  6. 释放LRU缓存:

    void lruCacheFree(LRUCache* cache) {
        CacheNode* current = cache->head;
        while (current != NULL) {
            CacheNode* next = current->next;
            free(current->key);
            free(current);
            current = next;
        }
        free(cache->table);
        free(cache);
    }

C语言LRU缓存实现中的哈希冲突如何处理?

lruCachePut函数中,当计算出的哈希索引对应的位置已经存在节点时,采用链地址法解决冲突。新的节点会以头插法的方式插入到哈希表对应索引的链表中。在lruCacheGet函数中,如果发生哈希冲突,会遍历链表,直到找到匹配的key。

EasySite
EasySite

零代码AI网站开发工具

下载

C语言实现LRU缓存的线程安全性问题

上述代码并非线程安全。在多线程环境下,对缓存的并发访问可能导致数据竞争和不一致。例如,多个线程同时尝试插入或删除节点,可能导致链表结构损坏或哈希表数据不一致。

为了实现线程安全,需要使用互斥锁(mutex)来保护对缓存数据结构的访问。

  1. 添加互斥锁:LRUCache结构体中添加一个互斥锁成员。

    typedef struct LRUCache {
        size_t capacity;
        size_t size;
        CacheNode *head;
        CacheNode *tail;
        CacheNode **table;
        pthread_mutex_t mutex; // 互斥锁
    } LRUCache;
  2. 初始化互斥锁:lruCacheCreate函数中初始化互斥锁。

    LRUCache* lruCacheCreate(size_t capacity) {
        // ... 其他初始化代码 ...
        pthread_mutex_init(&cache->mutex, NULL); // 初始化互斥锁
        return cache;
    }
  3. 加锁和解锁:lruCacheGetlruCachePutlruCacheFree函数中,在访问或修改缓存数据结构之前加锁,操作完成后解锁。

    void* lruCacheGet(LRUCache* cache, const char *key) {
        pthread_mutex_lock(&cache->mutex); // 加锁
        // ... 缓存访问代码 ...
        pthread_mutex_unlock(&cache->mutex); // 解锁
        return result;
    }
    
    void lruCachePut(LRUCache* cache, const char *key, void *value) {
        pthread_mutex_lock(&cache->mutex); // 加锁
        // ... 缓存修改代码 ...
        pthread_mutex_unlock(&cache->mutex); // 解锁
    }
    
    void lruCacheFree(LRUCache* cache) {
        pthread_mutex_lock(&cache->mutex); // 加锁
        // ... 缓存释放代码 ...
        pthread_mutex_unlock(&cache->mutex); // 解锁
        pthread_mutex_destroy(&cache->mutex); // 销毁互斥锁
        // ... 其他释放代码 ...
    }

如何选择合适的哈希函数以提高LRU缓存性能?

哈希函数的选择对LRU缓存的性能至关重要。一个好的哈希函数应该具有以下特点:

  • 均匀性: 哈希函数应该将不同的key均匀地映射到哈希表的各个槽位,避免出现大量的哈希冲突。
  • 高效性: 哈希函数的计算速度应该足够快,以减少缓存操作的延迟。
  • 简单性: 哈希函数的实现应该简单易懂,方便维护和调试。

一些常用的哈希函数包括:

  • MurmurHash: 一种非加密哈希函数,具有良好的均匀性和高效性。
  • FNV-1a: 另一种非加密哈希函数,实现简单,性能也不错。
  • DJB2: 一种经典的哈希函数,广泛应用于各种场景。

选择哈希函数时,需要根据具体的应用场景进行权衡。如果对性能要求非常高,可以考虑使用MurmurHash或FNV-1a。如果对实现简单性要求较高,可以使用DJB2。此外,还可以根据key的特点选择特定的哈希函数,例如,如果key是字符串,可以使用专门为字符串设计的哈希函数。

相关专题

更多
C语言变量命名
C语言变量命名

c语言变量名规则是:1、变量名以英文字母开头;2、变量名中的字母是区分大小写的;3、变量名不能是关键字;4、变量名中不能包含空格、标点符号和类型说明符。php中文网还提供c语言变量的相关下载、相关课程等内容,供大家免费下载使用。

397

2023.06.20

c语言入门自学零基础
c语言入门自学零基础

C语言是当代人学习及生活中的必备基础知识,应用十分广泛,本专题为大家c语言入门自学零基础的相关文章,以及相关课程,感兴趣的朋友千万不要错过了。

618

2023.07.25

c语言运算符的优先级顺序
c语言运算符的优先级顺序

c语言运算符的优先级顺序是括号运算符 > 一元运算符 > 算术运算符 > 移位运算符 > 关系运算符 > 位运算符 > 逻辑运算符 > 赋值运算符 > 逗号运算符。本专题为大家提供c语言运算符相关的各种文章、以及下载和课程。

354

2023.08.02

c语言数据结构
c语言数据结构

数据结构是指将数据按照一定的方式组织和存储的方法。它是计算机科学中的重要概念,用来描述和解决实际问题中的数据组织和处理问题。数据结构可以分为线性结构和非线性结构。线性结构包括数组、链表、堆栈和队列等,而非线性结构包括树和图等。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

258

2023.08.09

c语言random函数用法
c语言random函数用法

c语言random函数用法:1、random.random,随机生成(0,1)之间的浮点数;2、random.randint,随机生成在范围之内的整数,两个参数分别表示上限和下限;3、random.randrange,在指定范围内,按指定基数递增的集合中获得一个随机数;4、random.choice,从序列中随机抽选一个数;5、random.shuffle,随机排序。

600

2023.09.05

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

526

2023.09.20

c语言get函数的用法
c语言get函数的用法

get函数是一个用于从输入流中获取字符的函数。可以从键盘、文件或其他输入设备中读取字符,并将其存储在指定的变量中。本文介绍了get函数的用法以及一些相关的注意事项。希望这篇文章能够帮助你更好地理解和使用get函数 。

642

2023.09.20

c数组初始化的方法
c数组初始化的方法

c语言数组初始化的方法有直接赋值法、不完全初始化法、省略数组长度法和二维数组初始化法。详细介绍:1、直接赋值法,这种方法可以直接将数组的值进行初始化;2、不完全初始化法,。这种方法可以在一定程度上节省内存空间;3、省略数组长度法,这种方法可以让编译器自动计算数组的长度;4、二维数组初始化法等等。

601

2023.09.22

html编辑相关教程合集
html编辑相关教程合集

本专题整合了html编辑相关教程合集,阅读专题下面的文章了解更多详细内容。

16

2026.01.21

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Rust 教程
Rust 教程

共28课时 | 4.7万人学习

Kotlin 教程
Kotlin 教程

共23课时 | 2.7万人学习

Go 教程
Go 教程

共32课时 | 4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号