• 【leetcode热题Hot100】——LRU缓存


    • 💂 个人主页:努力学习的少年
    • 🤟 版权: 本文由【努力学习的少年】原创、在CSDN首发、需要转载请联系博主
    • 💬 如果文章对你有帮助、欢迎关注、点赞、收藏(一键三连)和订阅专栏哦

    目录

    一. 题目

    1.题目描述

    2.基础框架

    3.原题链接

    二.解题报告

    代码详解


    一. 题目

    1.题目描述

    请你设计并实现一个满足  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]

    2.基础框架

    1. class LRUCache {
    2. public:
    3. LRUCache(int capacity) {
    4. }
    5. int get(int key) {
    6. }
    7. void put(int key, int value) {
    8. }
    9. };

    3.原题链接

    LRU缓存

    二.解题报告

    思路:

    该结构我们将设置为 双向链表+哈希表 的结构。

    哈希表存储的 key-节点的地址,方便找到对应的节点。

    双向链表中的节点存储的是 key-value 的值,该双向链表包含了 head节点 tail节点,head节点指向的是最近被使用过的节点(新插入,修改,返回值),tail节点指向的是最久未使用的节点。

    void put(int key,int value)实现;

    • 如果key值在哈希表中找不到,则说明该key值的节点不存在,因此 我们需要将key值节点插入到head的后面

    如下:新插入4-4的节点,通过哈希表发现,该key值不存在,说明key=4的节点不存在。

    • 如果key值在哈希表总找到了,则说明该key值的节点存在,因此我们去修改节点的value,并将修改的节点链接到后面,代表该节点最近被使用过。

    如下:put(3,4);因为key=3节点已经存在,所以我们需要该节点的value进行修改。

    首先通过哈希表找到该节点,修改后再将该节点放在head的后面,表示该节点最近被使用过。

     

    • 如果插入的节点的数量大于capacity,我们需要删除链表最后一个节点,也就是tail节点前面的节点,在将新节点插入到head节点的后面。

    假设LRU的capacity为3,那么如果我们在插入一个key-value值为4,4,我们需要删除tail最后一个节点,因为它是最久没有被使用过的节点,然后将删除的节点在哈希表的映射关系。再将key=4的节点插入到head后面,如下

    int get(value)函数

    • 如果key值在哈希表中找不到,直接返回-1。
    • 如果找到了,那么找到相对应的节点,并将该节点连接到head的后面,代表该节点最近是被使用过,然后再返回它的value值。

    如下:查找key的值为2;

     我们将1节点和3节点互相连接,然后再将head与2节点互相连接,连接的时间复杂度为O(1).

    总结:由于key-节点的地址存储在哈希表中,因此查找节点的时间复杂度是O(1),节点连接的过程也是O(1),删除节点是直接将tail前面的节点删除,因此时间复杂度也为O(1),因此,整个get()和put()的时间复杂度都是O(1).

    代码详解

    1. //设置Node节点类
    2. class Node{
    3. public:
    4. int _key;
    5. int _value;
    6. Node* prev=nullptr;
    7. Node* next=nullptr;
    8. public:
    9. Node(int key=0,int value=0):_key(key),_value(value){}
    10. };
    11. class LRUCache {
    12. public:
    13. LRUCache(int capacity) {
    14. _capacity=capacity;
    15. head=new Node();
    16. tail=new Node();
    17. head->next=tail;
    18. tail->prev=head;
    19. }
    20. int get(int key) {
    21. //该key值没有在LRU缓存中
    22. if(mp.count(key)==0){
    23. return -1;
    24. }
    25. //该key值在LRU缓存中
    26. //找到该节点,连接前后节点
    27. Node* cur=mp[key];
    28. if(head->next==cur)
    29. return cur->_value;
    30. Node* next=cur->next;
    31. Node* prev=cur->prev;
    32. Node* headNext=head->next;
    33. next->prev=prev;
    34. prev->next=next;
    35. //将该节点与head互相连接
    36. head->next=cur;
    37. cur->prev=head;
    38. cur->next=headNext;
    39. headNext->prev=cur;
    40. return cur->_value;
    41. }
    42. void put(int key, int value) {
    43. if(mp.count(key)){
    44. //该节点存在,修改节点的value值
    45. Node* cur=mp[key];
    46. cur->_value=value;
    47. if(head->next==cur)
    48. return;
    49. Node* prev=cur->prev;
    50. Node* next=cur->next;
    51. Node* headNext=head->next;
    52. prev->next=next;
    53. next->prev=prev;
    54. head->next=cur;
    55. cur->prev=head;
    56. cur->next=headNext;
    57. headNext->prev=cur;
    58. return;
    59. }else{
    60. //不存在
    61. if(size==_capacity){
    62. //如果LRU缓存满了,删除最后一个节点
    63. Node* prev=tail->prev;
    64. Node* pprev=prev->prev;
    65. pprev->next=tail;
    66. tail->prev=pprev;
    67. mp.erase(prev->_key);
    68. delete prev;
    69. prev=nullptr;
    70. size--;
    71. }
    72. //创建新的节点,并将它连接在head的后面
    73. Node* cur=new Node(key,value);
    74. mp[key]=cur;
    75. Node* headNext=head->next;
    76. cur->prev=head;
    77. head->next=cur;
    78. cur->next=headNext;
    79. headNext->prev=cur;
    80. size++;
    81. }
    82. }
    83. private:
    84. Node* head;//双向链表的头节点
    85. Node* tail;//双向链表的末尾节点
    86. unordered_map<int,Node*> mp;//哈希表
    87. int _capacity;//LRU的容量
    88. int size=0;//LRU中存在节点的个数
    89. };

  • 相关阅读:
    前后端分离
    刷题5-合并两个有序数组
    分布式秒杀方案--java
    [VM trunk ports]opensatck VM 单网卡,多VLAN配置
    大数据库练习题目集-键值数据库-2022-2023-1-20大数据本
    Nvidia GPU 入门教程之 10 如何通过TensorFlow Datasets 下载海量数据集?
    可升级合约的原理-DelegateCall
    命令行远程操作windows
    计基2—RISCV指令集介绍与汇编
    RabbitMQ的使用
  • 原文地址:https://blog.csdn.net/sjp11/article/details/126021573