题目链接

LeetCode

题目描述

实现 LRUCache 类:

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

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

解题思路

方法一:哈希表 + 双向链表

LRU 缓存机制可以通过哈希表辅以双向链表实现,我们用一个哈希表和一个双向链表维护所有在缓存中的键值对。

  • 双向链表按照被使用的顺序存储了这些键值对,靠近头部的键值对是最近使用的,而靠近尾部的键值对是最久未使用的。
  • 哈希表即为普通的哈希映射(HashMap),通过缓存数据的键映射到其在双向链表中的位置。

这样以来,我们首先使用哈希表进行定位,找出缓存项在双向链表中的位置,随后将其移动到双向链表的头部,即可在 O(1)的时间内完成 get 或者 put 操作。具体的方法如下:

  • 对于 get 操作,首先判断 key 是否存在:
    • 如果 key 不存在,则返回 −1;
    • 如果 key 存在,则 key 对应的节点是最近被使用的节点。通过哈希表定位到该节点在双向链表中的位置,并将其移动到双向链表的头部,最后返回该节点的值。
  • 对于 put 操作,首先判断 key 是否存在:
    • 如果 key 不存在,使用 keyvalue 创建一个新的节点,在双向链表的头部添加该节点,并将 key 和该节点添加进哈希表中。然后判断双向链表的节点数是否超出容量,如果超出容量,则删除双向链表的尾部节点,并删除哈希表中对应的项;
    • 如果 key 存在,则与 get 操作类似,先通过哈希表定位,再将对应的节点的值更新为 value,并将该节点移到双向链表的头部。

上述各项操作中,访问哈希表的时间复杂度为 O(1),在双向链表的头部添加节点、在双向链表的尾部删除节点的复杂度也为 O(1)。而将一个节点移到双向链表的头部,可以分成「删除该节点」和「在双向链表的头部添加节点」两步操作,都可以在 O(1) 时间内完成。

小贴士

在双向链表的实现中,使用一个伪头部(dummy head)和伪尾部(dummy tail)标记界限,这样在添加节点和删除节点的时候就不需要检查相邻的节点是否存在。

146. LRU缓存机制** - 图1
146. LRU缓存机制** - 图2
146. LRU缓存机制** - 图3
146. LRU缓存机制** - 图4
146. LRU缓存机制** - 图5
146. LRU缓存机制** - 图6
146. LRU缓存机制** - 图7
146. LRU缓存机制** - 图8
146. LRU缓存机制** - 图9
146. LRU缓存机制** - 图10

146. LRU缓存机制** - 图11
146. LRU缓存机制** - 图12
146. LRU缓存机制** - 图13
146. LRU缓存机制** - 图14
146. LRU缓存机制** - 图15
146. LRU缓存机制** - 图16
146. LRU缓存机制** - 图17
146. LRU缓存机制** - 图18
146. LRU缓存机制** - 图19
146. LRU缓存机制** - 图20

146. LRU缓存机制** - 图21
146. LRU缓存机制** - 图22
146. LRU缓存机制** - 图23
146. LRU缓存机制** - 图24
146. LRU缓存机制** - 图25
146. LRU缓存机制** - 图26
146. LRU缓存机制** - 图27
146. LRU缓存机制** - 图28
146. LRU缓存机制** - 图29
146. LRU缓存机制** - 图30

代码:

  1. class LRUCache {
  2. struct DLinkedNode{
  3. int key;
  4. int value;
  5. DLinkedNode* prev;
  6. DLinkedNode* next;
  7. DLinkedNode(){
  8. key = 0;
  9. value = 0;
  10. prev = nullptr;
  11. next = nullptr;
  12. }
  13. DLinkedNode(int _key,int _value){
  14. key = _key;
  15. value = _value;
  16. prev = nullptr;
  17. next = nullptr;
  18. }
  19. };
  20. private:
  21. unordered_map<int,DLinkedNode*> cache;
  22. DLinkedNode* head;
  23. DLinkedNode* tail;
  24. int size;
  25. int capacity;
  26. public:
  27. LRUCache(int capacity) {
  28. this->size = 0;
  29. this->capacity = capacity;
  30. head = new DLinkedNode();
  31. tail = new DLinkedNode();
  32. head->next = tail;
  33. tail->prev = head;
  34. }
  35. int get(int key) {
  36. if(!cache.count(key)){
  37. return -1;
  38. }
  39. DLinkedNode* node = cache[key];
  40. moveToHead(node);
  41. return node->value;
  42. }
  43. void put(int key, int value) {
  44. if(!cache.count(key)){
  45. DLinkedNode* node = new DLinkedNode(key,value);
  46. cache[key] = node;
  47. addToHead(node);
  48. ++size;
  49. if(size>capacity){
  50. DLinkedNode* node = removeTail();
  51. cache.erase(node->key);
  52. delete node;
  53. --size;
  54. }
  55. }else{
  56. DLinkedNode* node = cache[key];
  57. node->value = value;
  58. moveToHead(node);
  59. }
  60. }
  61. // 下面的操作都是链表操作,没有删除结点内存
  62. void addToHead(DLinkedNode* node){
  63. node->prev = head;
  64. node->next = head->next;
  65. head->next->prev = node;
  66. head->next = node;
  67. }
  68. void removeNode(DLinkedNode* node){
  69. node->next->prev = node->prev;
  70. node->prev->next = node->next;
  71. }
  72. void moveToHead(DLinkedNode* node){
  73. removeNode(node);
  74. addToHead(node);
  75. }
  76. // 返回需要删除的结点,在put函数中删除内存
  77. DLinkedNode* removeTail(){
  78. DLinkedNode* node = tail->prev;
  79. removeNode(node);
  80. return node;
  81. }
  82. };
  83. /**
  84. * Your LRUCache object will be instantiated and called as such:
  85. * LRUCache* obj = new LRUCache(capacity);
  86. * int param_1 = obj->get(key);
  87. * obj->put(key,value);
  88. */

复杂度分析

  • 时间复杂度:对于 putget 都是 O(1)。
  • 空间复杂度:O(capacity)