数据结构---LRU CACHE

这篇具有很好参考价值的文章主要介绍了数据结构---LRU CACHE。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

什么是LRU

通过之前的学习我们知道计算机在处理任务的时候是先将数据从硬盘中提取出来加载进内存,然后再将内存中的数据加载进入cpu进行计算,但是这里存在一个问题cpu的计算速度非常快,而内存中加载数据的速度又很慢,所以为了提供整机的工作效率我们就得在内存和cpu计算机中添加一个东西叫做缓存,狭义的Cache指的是位于CPU和主存间的快速RAM(缓存),通常它不像系统主存那样使用DRAM技术,而使用昂贵但较快速的SRAM技术。 广义上的Cache指的是位于速度相差较大的两种硬件之间,用于协调两者数据传输速度差异的结构。除了CPU与主存之间有Cache,内存与硬盘之间也有Cache,乃至在硬盘与网络之间也有某种意义上的Cache── 称为Internet临时文件夹或网络内容缓存等。但是Cache的容量有限,因此当Cache的容量用完后,而又有新的内容需要添加进来时, 就需要挑选并舍弃原有的部分内容,从而腾出空间来放新内容。那么将哪部分数据删除这就由LRU来决定,设计一个LRU并不难,但是要想设计一个高效的LRU是很有难度的,这里的高效指的是任何操作的时间复杂度都是o(1), LRU Cache 的替换原则就是将最近最少使用的内容替换掉。其实,LRU译成最久未使用会更形象,因为该算法每次替换掉的就是一段时间内最久没有使用过的内容,那么接下来我们就通过一道题来带着大家模拟实现LRU,题目的内容如下:
数据结构---LRU CACHE,c++详解,数据结构,开发语言,c++,算法
题目给的代码如下:

class LRUCache {
public:
    LRUCache(int capacity) {

    }
    
    int get(int key) {

    }
    
    void put(int key, int value) {

    }
};

/**
 * Your LRUCache object will be instantiated and called as such:
 * LRUCache* obj = new LRUCache(capacity);
 * int param_1 = obj->get(key);
 * obj->put(key,value);
 */

模拟实现LRU

通过上面的介绍我们知道get函数就是到缓存里面查找数据,如果找到了就返回数据的内容如果没有找到就返回-1,然后put函数有两个功能一个是往容器里面插入数据,另外一个就是更新容器里面的数据,那么要想实现上述功能的话我们可以添加一个哈希表来实现,这样我们查找数据的时间复杂度是o(1),删除数据的时间复杂度也是o(1),更新数据的时间复杂度也是o(1),但是这么做存在一个问题这里的时间复杂度确实符合要求了,但是我们这里要实现的是LRU啊,在删除数据的时候你怎么知道谁的优先级高谁的优先级低呢?简单的说就是你怎么知道要删除的是哪些数据呢?所以单独的一个哈希表肯定是无法满足要求的,那么这时有人说那能不能添加一个优先级队列来辅助实现呢?答案也是不行的,优先级队列确实可以将数据的优先级进行排序,我们一下子就可以知道谁的优先级最高,谁的优先级最低,但是这里有个问题当我们更新数据的时候将优先级调制最高,那你如何将里面的数据优先级队列中的那个元素进行调整呢?即使能够调整那能不能保证时间复杂度是o(1)呢?所以这种方法肯定是不行的那么这个时候有小伙伴又会说能不能添加引用计数的形式来实现呢?答案是可以的但是这么做还是太麻烦了,我们有一个很简单的方法就是添加链表来进行辅助,那么这里的代码如下:

class LRUCache {
public:
    LRUCache(int capacity) {

    }
    
    int get(int key) {

    }
    
    void put(int key, int value) {

    }
private:
    unordered_map<int, int> _hashmap;
    list<pair<int,int>> _l1;
};

查找和新增可以通过哈希表来做到时间复杂度为O(1),在删除的时候可以根据链表中元素的位置来判断优先级的高低,如果对一个元素进行更新的话就将这个链表调整值头部,也就是说越位于链表头部的节点优先级越高,相反位于越位于链表尾部的节点优先级越低,在删除的时候优先删除链表尾部的数据,这么听好像很有道理但是更新的时间复杂度能够做到o(1)吗?好像不行对吧,因为链表不支持随机访问,所以我们得循环遍历从而找到要更新的节点,那么这一步时间复杂度就不是o(1)了更何况后面的步骤,所以我们这里得进行改进,哈希表可以帮助我们在o(1)的时间复杂度找到想要的数据,那要想更新的问题我们就得在哈希表找到数据的同时找到数据对应在链表上的位置,那么这里的做法就是在修改哈希表中记录的元素类型,由原来的int改为list类中的迭代器类型,那么这里的代码就如下:

unordered_map<int, list<pair<int,int>>::iterator> _hashmap;
list<pair<int,int>> _l1;

哈希表中的元素有一个元素记录着链表中相应元素的位置,这样我们就可以使用O(1)的时间找到哈希表中的元素还顺便找到了链表中的元素,构造函数里面提供了一个名为capacity的变量,表示这个容器能够存储数据的个数,那么为了方便我们后面删除数据,这里就再添加一个数据表示当然LRU的容量,并且再使用typedef来重命名依次减轻代码的长度,然后构造函数里面只用对容量变量进行初始化就行那么这里的代码就如下:

class LRUCache {
public:
typedef list<pair<int,int>>::iterator LiIter;
    LRUCache(int capacity) 
        :_capacity(capacity)
    {}
    
    int get(int key) {

    }
    
    void put(int key, int value) {

    }
private:
    unordered_map<int,LiIter> _hashmap;
    list<pair<int,int>> _l1;
    size_t _capacity;
};

get函数非常好实现,首先使用find函数来查找当前的数据在还是不在,那么这里我们可以使用find函数来实现,如果find函数的返回值为哈希表的end的话就说明当前数据不存在,如果不为end的话我们就先通过操作->来得到内部的第二个数据,但是这里的数据依然为一个指向链表的迭代器,所以这里还要通过操作符->来获取第二个数据才是我们想要的,那么这里的代码就下:

int get(int key) {
    auto ret=_hashmap.find(key);
    if(ret!=_hashmap.end())
    {
        return ret->second->second;
    }
    else
    {
        return -1;
    } 
}

但是这里并没有结束,因为我们查找了这里的数据,所以这个数据的优先级应该要被改变,也就是说将元素所在的链表位置进行该表,那么这里有两种方法第一种就是先记录当前的元素,然后将元素进行删除并在链表的头部插入一个相同的数据,最后修改迭代器的指向,但是这种方法太麻烦了我们可以直接使用容器中的splice函数来实现节点的转移,这个函数的参数形式和功能介绍如下:
数据结构---LRU CACHE,c++详解,数据结构,开发语言,c++,算法
数据结构---LRU CACHE,c++详解,数据结构,开发语言,c++,算法
我们可以将自己链表中的节点转移到自己链表的头部位置,那么这里的代码就如下:

int get(int key) {
    auto ret=_hashmap.find(key);
    if(ret!=_hashmap.end())
    {
        LiIter it =ret->second;
        _l1.splice(_l1.begin(),_l1,it);
        return it->second;
    }
    else
    {
        return -1;
    }
}

put函数分为两种情况:一个是新增一个是插入,我们首先来判断一下当前的数据在还是不在如果在的话就是新增,如果不在的话就是插入,那么这里可以通过find函数的返回值来进行判断:

void put(int key, int value) {
    auto ret=_hashmap.find(key);
    if(ret!=_hashmap.end())
    {
        //元素存在那么这里就是更新
    }
    else
    {
        //元素不存在这里是插入
    }
}

如果当前是插入的话这里得进行判断,但是这里判断的时候不能使用list的size函数而是得使用哈希的size函数,因为list中的size是顺序遍历时间复杂度为o(N),而哈希中的size是直接返回内部的数据,所以当对象满了之后我们就得删除数据,首先创建变量记录链表的尾部数据,然后使用哈希的erase函数删除该数据,最后使用pop_back函数删除链表的尾部数据,那么这里的代码如下:

void put(int key, int value) {
   auto ret=_hashmap.find(key);
   if(ret!=_hashmap.end())
   {
       //元素存在那么这里就是更新
   }
   else
   {
       //元素不存在这里是插入
       if(_capacity==_hashmap.size())
       {
           //满了就要删除
           pair<int,int> tmp=_l1.back();
           _l1.pop_back();
           _hashmap.erase(tmp.second);
       }
   }
}

将数据删除之后就往链表的头部插入数据,然后再往哈希表里面插入数据并将该元素的第二个元素初始化为容器开头的位置,那么这里的代码就如下:

void put(int key, int value) {
    auto ret=_hashmap.find(key);
    if(ret!=_hashmap.end())
    {
        //元素存在那么这里就是更新
    }
    else
    {
        //元素不存在这里是插入
        if(_capacity==_hashmap.size())
        {
            //满了就要删除
            pair<int,int> tmp=_l1.back();
            _l1.pop_back();
            _hashmap.erase(tmp.second);
        }
        _l1.push_front(make_pair(key,value));
        _hashmap[key]=_l1.begin();
    }
}

如果当前的对象没有满的话我们就要更新value的值和链表中当前元素所在的位置,那么这里的思路和前面的一致,只不过得修改一下存储的value即可,那么这里的代码如下:

    void put(int key, int value) {
        auto ret=_hashmap.find(key);
        if(ret!=_hashmap.end())
        {
            //元素存在那么这里就是更新
             LiIter it =ret->second;
             it->second=value;
            _l1.splice(_l1.begin(),_l1,it);
        }
        else
        {
            //元素不存在这里是插入
            if(_capacity==_hashmap.size())
            {
                //满了就要删除
                pair<int,int> tmp=_l1.back();
                _l1.pop_back();
                _hashmap.erase(tmp.second);
            }
            _l1.push_front(make_pair(key,value));
            _hashmap[key]=_l1.begin();
        }
    }

写到这里我们的代码就完成了,那么完整的代码如下:

class LRUCache {
public:
typedef list<pair<int,int>>::iterator LiIter;
    LRUCache(int capacity) 
        :_capacity(capacity)
    {}
    
    int get(int key) {
        auto ret=_hashmap.find(key);
        if(ret!=_hashmap.end())
        {
            LiIter it =ret->second;
            _l1.splice(_l1.begin(),_l1,it);
            return it->second;
        }
        else
        {
            return -1;
        }
    }
    
    void put(int key, int value) {
        auto ret=_hashmap.find(key);
        if(ret!=_hashmap.end())
        {
            //元素存在那么这里就是更新
             LiIter it =ret->second;
             it->second=value;
            _l1.splice(_l1.begin(),_l1,it);
        }
        else
        {
            //元素不存在这里是插入
            if(_capacity==_hashmap.size())
            {
                //满了就要删除
                pair<int,int> tmp=_l1.back();
                _l1.pop_back();
                _hashmap.erase(tmp.first);
            }
            _l1.push_front(make_pair(key,value));
            _hashmap[key]=_l1.begin();
        }
    }
private:
    unordered_map<int,LiIter> _hashmap;
    list<pair<int,int>> _l1;
    size_t _capacity;
};

题目测试的结果如下:
数据结构---LRU CACHE,c++详解,数据结构,开发语言,c++,算法
那么这就说明我们的代码没有问题,本篇文章到此结束。文章来源地址https://www.toymoban.com/news/detail-615489.html

到了这里,关于数据结构---LRU CACHE的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【算法与数据结构】 C语言实现单链表队列详解

    前面我们学习了队列的顺序表的实现,本节将用单链表实现队列。 队列也可以数组和链表的结构实现, 使用链表的结构实现更优一些,因为如果使用数组的结构,出队列在数组头上出数据,效率会比较低 。下面我们先复习一下队列的基本概念: 队列:只允许在一端进行插入

    2024年04月11日
    浏览(43)
  • 数据结构与算法教程,数据结构C语言版教程!(第五部分、数组和广义表详解)三

    数组和广义表,都用于存储逻辑关系为“一对一”的数据。 数组存储结构,99% 的编程语言都包含的存储结构,用于存储不可再分的单一数据;而广义表不同,它还可以存储子广义表。 本章重点从矩阵的角度讨论二维数组的存储,同时讲解广义表的存储结构以及有关其广度和

    2024年01月21日
    浏览(40)
  • 数据结构与算法教程,数据结构C语言版教程!(第五部分、数组和广义表详解)五

    数组和广义表,都用于存储逻辑关系为“一对一”的数据。 数组存储结构,99% 的编程语言都包含的存储结构,用于存储不可再分的单一数据;而广义表不同,它还可以存储子广义表。 本章重点从矩阵的角度讨论二维数组的存储,同时讲解广义表的存储结构以及有关其广度和

    2024年01月23日
    浏览(37)
  • 【数据结构和算法初阶(C语言)】复杂链表(随机指针,随机链表的复制)题目详解+链表顺序表结尾

    目录  1.随机链表的复制 1.2题目描述  1.3题目分析 1.4解题: 2.顺序表和链表对比 2.1cpu高速缓存利用率 3.结语 一个长度为  n  的链表,每个节点包含一个额外增加的随机指针  random   该指针可以指向链表中的任何节点或空节点。        构造这个链表的  深拷贝 。 深拷贝

    2024年03月10日
    浏览(60)
  • LRU数据结构

    LRU缓存是非常经典的数据结构设计,本文重点在于设计出get、put方法时间复杂度都为O(1)的LRU缓存 LRU缓存特征是当超出容量时会将最近最少使用的元素逐出 第一次的失败尝试——记录有效区间 记录有效区间的方法是通过计数器counter为每一个最近使用的元素记录最新计数,在

    2024年02月15日
    浏览(30)
  • Android开发UI新技能,你get这个新技能了吗?(附源码详解),从零开始学数据结构和算法

    2. 文本输入框 val state = +state { “Text Field to input” } TextField( value = state.value, onValueChange = { state.value = it } ) 3. 按钮 Button(text = “咬我啊”, onClick = { Log.v(“test”, “被咬了”) }) 4.弹出框 MaterialTheme { Column { val openDialog = +state { false } Button(“Click me”, onClick = { openDialog.value = true

    2024年04月12日
    浏览(26)
  • LRU 该用什么数据结构

    LRU(最近最少使用),是一种缓存置换算法。缓存是用来存储常用的数据,加速常用数据访问的数据结构。有软件实现,比如数据库的缓存;也有硬件实现,比如我们上一讲学的 TLB。 缓存设计中有一个重要的环节:当缓存满了,新的缓存条目要写入时,哪个旧条目被置换出

    2024年02月06日
    浏览(28)
  • 【数据结构】LRU缓存的简单模拟实现(leetcode力扣146LRU缓存)

    LRU是Least Recently Used的缩写,意思是最近最少使用,它是一种Cache替换算法。 Cache的容量有限,因此当Cache的容量用完后,而又有新的内容需要添加进来时, 就需要挑选并舍弃原有的部分内容,从而腾出空间来放新内容。LRU Cache 的替换原则就是将最近最少使用的内容替换掉。

    2024年02月03日
    浏览(34)
  • 【数据结构】C语言结构体详解

    目录 前言 一、结构体的定义 二、定义结构体变量 三、结构体变量的初始化 四、使用typedef声明新数据类型名 五、指向结构体变量的指针 总结 🌈嗨!我是Filotimo__🌈。很高兴与大家相识,希望我的博客能对你有所帮助。 💡本文由Filotimo__✍️原创,首发于CSDN📚。 📣如需转

    2024年02月04日
    浏览(40)
  • 数据结构与算法——数据结构有哪些,常用数据结构详解

    数据结构是学习数据存储方式的一门学科,那么,数据存储方式有哪几种呢?下面将对数据结构的学习内容做一个简要的总结。 数据结构大致包含以下几种存储结构: 线性表,还可细分为顺序表、链表、栈和队列; 树结构,包括普通树,二叉树,线索二叉树等; 图存储结构

    2024年02月15日
    浏览(45)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包