Algorithm-Pattern
  • Introduction
  • C & C++
    • C语言
      • C/C++编译器
      • 宏的使用
      • 编译过程
      • 指针 & 数组
      • 柔性数组
      • 函数指针 & 回调函数
      • C标准库之stdio
      • C标准库之string
      • C标准库之errno
      • C标准库之stdarg
      • C标准库之regex
    • C++基础语法
      • 自增(++) & 自减(--)
      • c语言到c++
      • 可变模板参数
      • 强制类型转换
      • C/C++类型转换的本质
      • 指针 & 引用
      • const的用法
      • static的用法
      • 重要的关键字(一)
      • 重要的关键字(二)
      • 内存申请和释放
      • 内联函数
      • 函数 & 运算符重载
      • 面向对象之封装
      • 构造函数 & 析构函数
      • 面向对象之继承
      • 面向对象之多态
      • 泛型编程
      • 异常
      • 再谈函数指针
    • C++并发编程
      • C++的锁
      • 并发与多线程
    • C++高级特性
      • 函数对象
      • 移动语义 & 完美转发
      • lambda表达式
      • RTTI技术
      • RAII技术
      • 智能指针
      • 模板的特化
      • C++静态库和动态库
      • 内存溢出和内存泄漏
    • STL基础
      • String
      • array/vector/list
      • deque/priority_queue
      • set/map
      • unordered_set/unordered_map
      • algorithm_1
      • functional
      • allocator
    • C++标准库
      • IO
      • Tuple
      • regex
      • bitset
      • numeric
    • STL深入源码
      • vector内部实现
      • deque内部实现
      • sort函数实现
    • 第三方库
      • JsonCpp
      • ProtoBuf
  • 数据结构
    • 线性表
    • 字符串
    • 栈和队列
    • 二叉树
    • 平衡二叉树
    • 平衡多路搜索树
    • 树结构的延申
    • 图
    • 二进制
    • 散列表
  • 算法基础
    • 排序算法
    • 查找算法
    • 数学问题
    • 并查集
    • 递归算法
    • 附加——主定理
    • Catalan数
  • 算法设计思想
    • 滑动窗口思想
    • BFS/DFS
    • 二分法
    • 回溯法
    • 贪心算法
    • 分治法
    • 动态规划
    • 分支限界算法
    • 有限状态机(FSM)
  • LeetCode系列
    • 死磕二叉树
    • 股票买卖问题
    • 打家劫舍问题
    • 跳跃游戏问题
    • 括号匹配问题
    • 石子游戏问题
    • 子序列问题
    • 数组 & 矩阵
    • 排列 & 组合
  • 经典算法问题
    • 几何问题
    • 区间问题
    • 背包问题
    • 石子堆问题
    • 表达式求值
  • 面试题
    • 数据结构和算法基础
    • 程序设计题
      • 实现双线程交替打印
      • C++实现读写锁
      • 实现阻塞队列
      • 实现环形队列
      • 实现线程池
      • 实现智能指针
      • 实现string类
      • 实现高性能local cache
      • 实现内存池
      • 生产者-消费者模型
      • 设计定时器
    • 经典的算法题
    • C++面试题总结
    • 面试算法题总结
由 GitBook 提供支持
在本页
  • 1、缓存机制
  • LRU Cache
  • LFU Cache
  • 2、实现local cache

这有帮助吗?

  1. 面试题
  2. 程序设计题

实现高性能local cache

上一页实现string类下一页实现内存池

最后更新于4年前

这有帮助吗?

1、缓存机制

设计和实现一个 LRU (最近最少使用) 缓存机制。它应该支持以下操作: 获取数据 get 和 写入数据 put 。

获取数据 get(key) - 如果关键字 (key) 存在于缓存中,则获取关键字的值(总是正数),否则返回 -1。 写入数据 put(key, value) - 如果关键字已经存在,则变更其数据值;如果关键字不存在,则插入该组「关键字/值」。当缓存容量达到上限时,它应该在写入新数据之前删除最久未使用的数据值,从而为新的数据值留出空间。

实现本题的两种操作,需要用到一个哈希表(快速查找)和一个双向链表(维持有序性)。在面试中,面试官一般会期望读者能够自己实现一个简单的双向链表,而不是使用语言自带的、封装好的数据结构。在 Python 语言中,有一种结合了哈希表与双向链表的数据结构 OrderedDict,只需要短短的几行代码就可以完成本题。在 Java 语言中,同样有类似的数据结构 LinkedHashMap。

设计并实现 最不经常使用(LFU)缓存算法数据结构。它应该支持以下操作:get 和 put。

get(key) - 如果键存在于缓存中,则获取键的值(总是正数),否则返回 -1。 put(key, value) - 如果键已存在,则变更其值;如果键不存在,请插入键值对。当缓存达到其容量时,则应该在插入新项之前,使最不经常使用的项无效。在此问题中,当存在平局(即两个或更多个键具有相同使用频率)时,应该去除最久未使用的键。 「项的使用次数」就是自插入该项以来对其调用 get 和 put 函数的次数之和。使用次数会在对应项被移除后置为 0 。

struct DLinkedList{
    int key,val;
    DLinkedList *prev;
    DLinkedList *next;
    DLinkedList() : key(0), val(0), prev(nullptr), next(nullptr){}
    DLinkedList(int _key, int _val) : key(_key), val(_val), prev(nullptr), next(nullptr){}
};

class LRUCache {
public:
    LRUCache(int capacity) : m_capacity(capacity) , m_size(0){
        head = new DLinkedList();
        tail = new DLinkedList();
        head->next = tail;
        tail->prev = head;
    }
    int get(int key) {
        if(!m_cache.count(key)){
            return -1;
        }
        DLinkedList *node = m_cache[key];
        moveToHead(node);
        return node->val;
    }
    void put(int key, int value) {
        if(m_cache.count(key)){
            DLinkedList *node = m_cache[key];
            node->val = value;
            moveToHead(node);
        }else{
            DLinkedList *node = new DLinkedList(key, value);
            m_cache[key] = node;
            addToHead(node);
            ++m_size;
            if(m_size > m_capacity){
                DLinkedList *removeNode = removeTail();
                m_cache.erase(removeNode->key);
                delete removeNode;
                --m_size;
            }
        }
    }
    // 维护双向链表
    void moveToHead(DLinkedList *_node){
        removeNode(_node);
        addToHead(_node);
    }
    DLinkedList *removeTail(){
        DLinkedList *node = tail->prev;
        removeNode(node);
        return node;
    }

    void removeNode(DLinkedList *node){
        node->prev->next = node->next;
        node->next->prev = node->prev;
    }
    void addToHead(DLinkedList *node){
        node->next = head->next;
        head->next->prev = node;
        head->next = node;
        node->prev = head;
    }

private:
    int m_capacity;
    int m_size;
    DLinkedList *head;
    DLinkedList *tail;
    std::unordered_map<int, DLinkedList*> m_cache;
};

2、实现local cache

✏️
✏️
LRU Cache
LFU Cache