• ​LeetCode解法汇总146. LRU 缓存


     目录链接:

    力扣编程题-解法汇总_分享+记录-CSDN博客

    GitHub同步刷题项目:

    https://github.com/September26/java-algorithms

    原题链接:力扣(LeetCode)官网 - 全球极客挚爱的技术成长平台


    描述:

    请你设计并实现一个满足  LRU (最近最少使用) 缓存 约束的数据结构

    实现 LRUCache 类:

    • LRUCache(int capacity) 以 正整数 作为容量 capacity 初始化 LRU 缓存
    • int get(int key) 如果关键字 key 存在于缓存中,则返回关键字的值,否则返回 -1 。
    • void put(int key, int value) 如果关键字 key 已经存在,则变更其数据值 value ;如果不存在,则向缓存中插入该组 key-value 。如果插入操作导致关键字数量超过 capacity ,则应该 逐出 最久未使用的关键字。

    函数 get 和 put 必须以 O(1) 的平均时间复杂度运行。

    示例:

    输入
    ["LRUCache", "put", "put", "get", "put", "get", "put", "get", "get", "get"]
    [[2], [1, 1], [2, 2], [1], [3, 3], [2], [4, 4], [1], [3], [4]]
    输出
    [null, null, null, 1, null, -1, null, -1, 3, 4]
    
    解释
    LRUCache lRUCache = new LRUCache(2);
    lRUCache.put(1, 1); // 缓存是 {1=1}
    lRUCache.put(2, 2); // 缓存是 {1=1, 2=2}
    lRUCache.get(1);    // 返回 1
    lRUCache.put(3, 3); // 该操作会使得关键字 2 作废,缓存是 {1=1, 3=3}
    lRUCache.get(2);    // 返回 -1 (未找到)
    lRUCache.put(4, 4); // 该操作会使得关键字 1 作废,缓存是 {4=4, 3=3}
    lRUCache.get(1);    // 返回 -1 (未找到)
    lRUCache.get(3);    // 返回 3
    lRUCache.get(4);    // 返回 4
    

    提示:

    • 1 <= capacity <= 3000
    • 0 <= key <= 10000
    • 0 <= value <= 105
    • 最多调用 2 * 105 次 get 和 put

    解题思路:

    * 解题思路:

    * 这题最难的其实就是处理双向链表的关系。

    * 构建两个方法updateHead和updateTail,分别代表节点变动后处理头部节点和尾部节点。

    * 处理头部节点时,分为两种情况,当前节点如果已经是头部节点,则不需要处理。否则,把current的前后节点相连接,然后把current放到头节点。

    * 处理尾部节点时,分为3种情况,如果map中只有一个,则当前节点就是尾节点;如果map长度为2,则之前的header就是尾节点;如果current是尾节点,则断开其与前面节点的关系,设置其前面的节点为尾节点。

    代码:

    1. class LRUCache
    2. {
    3. public:
    4. class Node
    5. {
    6. public:
    7. int key;
    8. int val;
    9. Node *next;
    10. Node *pre;
    11. Node(int mkey = 0, int mvalue = 0) : key(mkey), val(mvalue), next(nullptr), pre(nullptr){};
    12. };
    13. int maxSize = 0;
    14. unordered_map<int, Node *> valueMap;
    15. Node *header = nullptr;
    16. Node *tail = nullptr;
    17. LRUCache(int capacity)
    18. {
    19. maxSize = capacity;
    20. }
    21. int get(int key)
    22. {
    23. if (valueMap.find(key) == valueMap.end())
    24. {
    25. return -1;
    26. }
    27. Node *current = valueMap[key];
    28. updateTail(current);
    29. updateHead(current);
    30. return valueMap[key]->val;
    31. }
    32. void put(int key, int value)
    33. {
    34. if (valueMap.find(key) != valueMap.end())
    35. {
    36. Node *current = valueMap[key];
    37. current->val = value;
    38. updateTail(current);
    39. updateHead(current);
    40. return;
    41. }
    42. if (valueMap.size() == maxSize)
    43. {
    44. removeNode();
    45. }
    46. addNode(key, value);
    47. }
    48. void addNode(int key, int value)
    49. {
    50. Node *current = new Node(key, value);
    51. valueMap[key] = current;
    52. updateTail(current);
    53. updateHead(current);
    54. }
    55. /**
    56. * 把current设置为header
    57. */
    58. void updateHead(Node *current)
    59. {
    60. if (header == current)
    61. {
    62. return;
    63. }
    64. // 链表中删除当前节点
    65. if (current->pre != nullptr)
    66. {
    67. current->pre->next = current->next;
    68. }
    69. if (current->next != nullptr)
    70. {
    71. current->next->pre = current->pre;
    72. }
    73. // 加入头节点
    74. current->next = header;
    75. current->pre = nullptr;
    76. if (header != nullptr)
    77. {
    78. header->pre = current;
    79. }
    80. header = current;
    81. }
    82. void updateTail(Node *current)
    83. {
    84. // 如果长度为1
    85. if (valueMap.size() == 1)
    86. {
    87. tail = current;
    88. return;
    89. }
    90. if (valueMap.size() == 2)
    91. {
    92. tail->pre = current;
    93. tail = header;
    94. }
    95. else if (current == tail)
    96. {
    97. if (tail->pre != nullptr)
    98. {
    99. tail->pre->next = nullptr;
    100. }
    101. if (valueMap.size() > 1)
    102. {
    103. tail = current->pre;
    104. }
    105. }
    106. }
    107. void removeNode()
    108. {
    109. valueMap.erase(tail->key);
    110. Node *tailPre = tail->pre;
    111. if (tailPre == nullptr)
    112. {
    113. return;
    114. }
    115. tailPre->next = nullptr;
    116. tail->pre = nullptr;
    117. tail = tailPre;
    118. }
    119. };

  • 相关阅读:
    Milvus+Attu
    springboot logback 日志注入安全问题 统一处理
    Tomcat自启动另一种方法
    SpringCloud Alibaba【二】nacos
    react之生命周期
    Kubernetes面试题分享
    分享记账软件 设置账本密码防止收支被他人修改
    计算机网络-网络层:IP协议
    python正则表达式的使用
    物联网通信消息队列客户端-MQTT简介
  • 原文地址:https://blog.csdn.net/AA5279AA/article/details/133272276