zoukankan      html  css  js  c++  java
  • LeetCode146 LRU缓存机制

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

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

    进阶:

    你是否可以在 O(1) 时间复杂度内完成这两种操作?

    使用一个哈希表记录key,用双链表记录key,value和顺序。哈希表中key对应的是双链表节点的指针。每次查询就把当前节点移动到链表的头部。每次put的时候也会把put的节点移动到头部,同时判断是否超出capacity,如果超出就删除dummyhead前面的一个节点。

    注意delete释放空间,修改指针要修改全。

    链表双向主要是可以在O(1)时间内找到需要删除的节点(即dummyTail前一个节点)。节点需要记录key主要是删除的时候得到key,这样才能在O(1)时间内在哈希表中找到并删除。

     1 struct DLinkedNode {
     2     int key, value;
     3     DLinkedNode* prev;
     4     DLinkedNode* next;
     5     DLinkedNode() : key(0), value(0), prev(nullptr), next(nullptr) {}
     6     DLinkedNode(int _key, int _value) : key(_key), value(_value), prev(nullptr), next(nullptr) {}
     7 };
     8 
     9 struct LRU_Node {
    10     int key;
    11     int value;
    12     LRU_Node* next;
    13     LRU_Node* prev;
    14     LRU_Node() : key(0), value(0), next(nullptr), prev(nullptr) {}
    15     LRU_Node(int _key, int _value) :key(_key), value(_value), next(nullptr), prev(nullptr) {}
    16 };
    17 
    18 class LRUCache {
    19 private:
    20     int num;
    21     int capacity;
    22     unordered_map<int, LRU_Node*> map_LRU;
    23     LRU_Node* dummyHead;
    24     LRU_Node* dummyTail;
    25 
    26 public:
    27     LRUCache(int _capacity) {
    28         num = 0;
    29         capacity = _capacity;
    30         dummyHead = new LRU_Node(-1, -1);
    31         dummyTail = new LRU_Node(-1, -1);
    32         dummyHead->next = dummyTail;
    33         dummyTail->prev = dummyHead;
    34         map_LRU.clear();
    35         cout << "LRUCache is created.
    ";
    36     }
    37 
    38     LRU_Node* moveToHead(LRU_Node* cur) {
    39         cur->prev->next = cur->next;
    40         cur->next->prev = cur->prev;
    41         cur->next = dummyHead->next;
    42         cur->prev = dummyHead;
    43         dummyHead->next = cur;
    44         cur->next->prev = cur;
    45         return cur;
    46     }
    47 
    48     LRU_Node* insertHead(LRU_Node* new_node) {
    49         new_node->next = dummyHead->next;
    50         new_node->prev = dummyHead;
    51         dummyHead->next = new_node;
    52         new_node->next->prev = new_node;
    53         return new_node;
    54     }
    55 
    56     int get(int key) {
    57         if (map_LRU.find(key) != map_LRU.end()) {
    58             LRU_Node* cur = map_LRU[key];
    59             cur = moveToHead(cur);
    60             map_LRU[key] = cur;
    61             return cur->value;
    62         }
    63         return -1;
    64     }
    65 
    66     void put(int key, int value) {
    67         if (map_LRU.find(key) != map_LRU.end()) {
    68             LRU_Node* cur = map_LRU[key];
    69             cur->value = value;
    70             cur = moveToHead(cur);
    71             map_LRU[key] = cur;
    72         }
    73         else {
    74             LRU_Node* new_node = new LRU_Node(key, value);
    75             new_node = insertHead(new_node);
    76             map_LRU[key] = new_node;
    77             if (num != capacity)
    78                 ++num;
    79             else {
    80                 int deleteKey = dummyTail->prev->key;
    81                 LRU_Node* toBeDelete=dummyTail->prev;
    82                 dummyTail->prev=dummyTail->prev->prev;
    83                 dummyTail->prev->next=dummyTail;
    84                 delete toBeDelete;
    85                 toBeDelete=nullptr;
    86                 auto it = map_LRU.find(deleteKey);
    87                 map_LRU.erase(it);
    88             }
    89         }
    90         return;
    91     }
    92 };
    93 
    94 /**
    95  * Your LRUCache object will be instantiated and called as such:
    96  * LRUCache* obj = new LRUCache(capacity);
    97  * int param_1 = obj->get(key);
    98  * obj->put(key,value);
    99  */
  • 相关阅读:
    理解OpenShift(5):从 Docker Volume 到 OpenShift Persistent Volume
    理解OpenShift(4):用户及权限管理
    理解OpenShift(3):网络之 SDN
    理解OpenShift(2):网络之 DNS(域名服务)
    理解OpenShift(1):网络之 Router 和 Route
    HTML盒子模型
    架构系统的雪崩理解
    C++11 lambda表达式学习
    C++11 std::shared_ptr总结与使用
    Kafka学习笔记
  • 原文地址:https://www.cnblogs.com/rookiez/p/13302297.html
Copyright © 2011-2022 走看看