用线性表加哈希表实现LRU
线性表+哈希表>>LRU算法
方法:哈希表 + 双向链表
算法
LRU 缓存机制可以通过哈希表辅以双向链表实现,我们用一个哈希表和一个双向链表维护所有在缓存中的键值对。
双向链表按照被使用的顺序存储了这些键值对,靠近头部的键值对是最近使用的,而靠近尾部的键值对是最久未使用的。
哈希表即为普通的哈希映射(HashMap),通过缓存数据的键映射到其在双向链表中的位置。
这样以来,我们首先使用哈希表进行定位,找出缓存项在双向链表中的位置,随后将其移动到双向链表的头部,即可在 O(1)O(1)O(1) 的时间内完成 get 或者 put 操作。具体的方法如下:
1 2 3 4 5
   | int LRU::get(int key)
  {
  }
   | 
 
对于 get 操作,首先判断 key 是否存在:
如果 key 不存在,则返回 −1;
如果 key 存在,则 key 对应的节点是最近被使用的节点。通过哈希表定位到该节点在双向链表中的位置,并将其移动到双向链表的头部,最后返回该节点的值。
对于 put 操作,首先判断 key 是否存在:
1 2 3 4
   | void LRU::put(int key,int value) {
  }
   | 
 
如果 key 不存在,使用 key 和 value 创建一个新的节点,在双向链表的头部添加该节点,并将 key 和该节点添加进哈希表中。然后判断双向链表的节点数是否超出容量,如果超出容量,则删除双向链表的尾部节点,并删除哈希表中对应的项;
如果 key 存在,则与 get 操作类似,先通过哈希表定位,再将对应的节点的值更新为 value,并将该节点移到双向链表的头部。
上述各项操作中,访问哈希表的时间复杂度为 O(1),在双向链表的头部添加节点、在双向链表的尾部删除节点的复杂度也为 O(1)。而将一个节点移到双向链表的头部,可以分成「删除该节点」和「在双向链表的头部添加节点」两步操作,都可以在 O(1)时间内完成。
小贴士
在双向链表的实现中,使用一个伪头部(dummy head)和伪尾部(dummy tail)标记界限,这样在添加节点和删除节点的时候就不需要检查相邻的节点是否存在。
复杂度分析
时间复杂度:对于 put 和 get 都是 O(1)。
空间复杂度:O(capacity)O(\text{capacity})O(capacity),因为哈希表和双向链表最多存储 capacity+1\text{capacity} + 1capacity+1 个元素。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104
   | #include <iostream> #include<list> #include<unordered_map> using namespace std; using std::cout; using std::endl; class LRU{ public:     LRU(int cap)     :_capacity(cap)     {
          cout << "LRU(int cap)" << endl;              }
      int get(int key);     void put(int key,int value);
  private:     struct cacheNode     {         cacheNode(int key,int v)         :_key(key)         ,_value(v)         {
              cout << "cacheNode(int key,int v)" << endl;
          }         int _key;         int _value;
      };
      list<cacheNode> _nodes;     int _capacity;     unordered_map<int,list<cacheNode>::iterator > _cache;     
  }; int LRU::get(int key) {
                     auto it = _cache.find(key);                                  if(it!=_cache.end())     {         _nodes.splice(_nodes.begin(),_nodes,it->second);                           return it->second->_value;     }     else     {         return -1;     }
  } void LRU::put(int key,int value) {                         auto it = _cache.find(key);      if(it!=_cache.end())     {         it->second->_value= value;         _nodes.splice(_nodes.begin(),_nodes,it->second);     }     else     {         if((int)_nodes.size()==_capacity)         {             auto &deleteNode = _nodes.back();             _cache.erase(deleteNode._key);                                                      _nodes.pop_back();         }         _nodes.push_front(cacheNode(key,value));         _cache.insert(std::make_pair(key,_nodes.begin()));     } }
  void test0(){
           LRU lru(2);     lru.put(1,88);     cout << "get(1)" << lru.get(1) << endl;     lru.put(3,99);     lru.put(4,77);     cout << "get(1)" << lru.get(1) << endl;
  }
  int main(void){     test0();     return 0; }
   |