力扣hot100:146. LRU 缓存

力扣hot100:146. LRU 缓存
在这里插入图片描述
听说华为实习笔试考了这题

如何使得插入操作时 O ( 1 ) O(1) O(1)呢?我们需要维护一个时间的长短,以便于取出离现在最长的时间,这个时间比较容易实现,我们维护一个time表示当前时间,从0开始,然后在使用的一些关键字里面,当一个关键字使用的时间越小,那么它越久未被使用,我们只需要维护一个关键字使用的最小值就行,并且还需要有更新操作。

  1. 如果我们使用优先队列,那么插入一次需要 O ( l o g n ) O(logn) O(logn),不满足要求
  2. 如果我们排序,那么排序一次需要 O ( n l o g n ) O(nlogn) O(nlogn),不满足要求
  3. 我们使用双向链表维护时间递增序列呢?使用哈希表快速查找结点
    • 答案是可行的!因为time是递增的,我们更新中间结点的时间,则必然会将这个结点移动到链表末尾,因此是 O ( 1 ) O(1) O(1)的。这样做我们甚至可以连时间都不用保存了

由于结点和值都是通过key来查询的,而查询只需要判断key,因此我们可以只使用一个哈希表,来同时维护这两个信息。
在这里插入图片描述

class LRUCache {
public:LRUCache(int capacity) {this->capacity = capacity;dummy = new List;end = dummy;}int get(int key) {if(mp.count(key) == 1){put_to_end(key);return mp[key].first;}return -1;}void put(int key, int value) {if(mp.count(key) == 1){mp[key].first = value;put_to_end(key);return;}//判断是否超出容量if(size == capacity){List * temp = dummy->next;dummy->next = temp->next;if(temp->next) temp->next->pre = dummy;mp.erase(temp->key);if(temp == end) end = temp->pre;delete temp;}else ++size;//插入全新元素List * lst = new List;lst->key = key;mp[key] = {value,lst};//放到末尾lst->pre = end;end->next = lst;end = lst;return;}
private:struct List{List * pre = nullptr;List * next = nullptr;int key;//该key用于删除头结点};void put_to_end(int key){List * access = mp[key].second;if(access == end) return;access->pre->next = access->next;if(access->next) access->next->pre = access->pre;access->next = nullptr;access->pre = end;end->next = access;end = access;return;}
private:unordered_map<int,pair<int,List *>> mp;int capacity;int size = 0;List * dummy;List * end;
};

需要注意的点:

  1. 容量为1,需要删除的结点可能就在队列末尾,因此需要和end比较
  2. 最新被使用的结点可能就在队列末尾,因此需要和end比较
  3. 更新新结点时,注意end需要指向它
  4. 除了使用伪首部之外,还可以使用伪尾部!这样对尾部的判断也减少了许多。

写完之后可以再思考一个,LRU:

  • LRU是一个最近最少使用的缓存机制,实现时,本质上,我们将最近使用的放到了双向链表的尾部,而最久没有使用的就再链表头,一旦我们需要这个值的时候,我们只需要使用哈希表快速查找到值在双向链表中的位置,再把它放到链表尾部即可。
  • 实际上我们这相当于维护了一个单调序列,并且能够快速修改它的值并更新位置,我们每次只取头部使用,这似乎比优先队列还快。但实际上他们的使用场景不一样,思考一下可以发现双向链表维护的单调序列,每次更新内部结点的值后它必然被更新到最大值,会被放在最后面,而不会更新后还在中间,如果更新后的值并不一定是最值,则这种方法就不适用了。

既然这是一个面试常考的题,那么写完后可以记住:LRU 缓存机制可以通过哈希表辅以双向链表实现,我们用一个哈希表和一个双向链表维护所有在缓存中的键值对。

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/16864.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

kettle学习之子映射组件

映射组件就跟java中的函数方法一样&#xff0c;类似一个子流程。 练习开始 根据数据库表中的id查询出想要的字段&#xff0c;并把字段存到excel表中 一、表输入 二、子映射 映射输入规范&#xff0c;类似java方法中的形参 name vsxcd是方法返回的参数 三、excel输出 运行结果…

VS2022编译CMake的工程

开源项目大都是用Make文件组织项目代码编译。对熟悉Window体系&#xff0c;一直用VS套件工作的人&#xff0c;还是有不小的隔阂。 好在有大神们帮助我们解决此类问题&#xff0c;使用CMake工具&#xff0c;可以自动转换工程类型。 1、解压缩代码&#xff0c;找到CMakeList.tx…

Python考试复习--day3

1.统计字符串个数 ninput() z0 s0 k0 o0 for i in n:if i.isalpha():zz1elif i.isnumeric():ss1elif i.isspace():k1else:o1 print(字母有{}个,数字有{}个,空格有{}个,其他字符{}个.format(z,s,k,o))2.分类统计字符 ninput() x0 d0 s0 k0 o0 for i in n:if i.islower():x1elif …

韩愈,文起八代之衰的儒学巨匠

&#x1f4a1; 如果想阅读最新的文章&#xff0c;或者有技术问题需要交流和沟通&#xff0c;可搜索并关注微信公众号“希望睿智”。 韩愈&#xff0c;字退之&#xff0c;生于唐代宗大历三年&#xff08;公元768年&#xff09;&#xff0c;卒于唐穆宗长庆四年&#xff08;公元82…

武汉网红餐馆火灾背后的安全警示:可燃气体报警器需定期校准

在餐饮业快速发展的今天&#xff0c;安全问题一直是行业内外关注的重点。 最近&#xff0c;武汉一家网红餐馆在就餐高峰期突发火灾&#xff0c;事件迅速成为公众关注的焦点。这一事故不仅给餐馆带来了重大损失&#xff0c;也引发了对于餐馆安全管理的深思。 尤其是可燃气体报…

基于SqlSugar的开发框架循序渐进介绍(20)-- 在基于UniApp+Vue的移动端实现多条件查询的处理

在做一些常规应用的时候&#xff0c;我们往往需要确定条件的内容&#xff0c;以便在后台进行区分的进行精确查询&#xff0c;在移动端&#xff0c;由于受限于屏幕界面的情况&#xff0c;一般会对多个指定的条件进行模糊的搜索&#xff0c;而这个搜索的处理&#xff0c;也是和前…

关于亚马逊、速卖通、虾皮、Lazada等平台自养号测评IP的重要性

在自养号测评中&#xff0c;IP的纯净度是一个至关重要的问题&#xff0c;它直接关系到账号的安全性和稳定性如果使用了被平台识别为异常或存在风险的IP地址&#xff0c;那么账号可能会面临被封禁的风险。这将对账号的正常使用和测评过程中造成严重影响。而使用纯净的IP地址&…

使用 Django ORM 进行数据库操作

文章目录 创建Django项目和应用定义模型查询数据更新和删除数据总结与进阶聚合和注解跨模型查询原始SQL查询 Django是一个流行的Web应用程序框架&#xff0c;它提供了一个强大且易于使用的对象关系映射&#xff08;ORM&#xff09;工具&#xff0c;用于与数据库进行交互。在本文…

教育新篇章:AI工具Sora引领学习新趋势

Sora——这个让人在24年初引爆AI圈的新产品&#xff0c;它究竟会如何改变我们的教育领域呢&#xff1f; 从gpt到Sora&#xff0c;从对话型的ai到游戏和短剧制作的新风口&#xff0c;我们从23年到24年一个接一个地被震惊&#xff01; Sora能够根据文本提示生成高质量的视频内容…

当 GIS 遇上 AI 大模型

今年整个 IT 界、甚至科技界最火爆的技术就是 AI 大模型了&#xff0c;没有之一。 以 OpenAI 的 GPT 为代表、国内外一众大模型跟进&#xff0c;形成了百模齐奔的态势。围绕着大模型&#xff0c;各项周边的应用也在快速发展。在 2023 年年底的云栖大会上&#xff0c;论坛标题带…

一个基于预训练的DenseNet121模型的人脸年龄分类系统

这篇文章采用预训练的DenseNet121模型并使用自定义的数据集类和自定义的类似正态分布的标签平滑策略来训练了一个人脸年龄分类模型&#xff0c;最后基于这个模型用tk实现了一个娱乐向的小系统。 数据集展示&#xff1a; 两个文件夹&#xff0c;分别是训练集和测试集&#xff0…

Bootstrap 3.x 版本基础引入指南

Bootstrap 是一款广受欢迎的前端框架&#xff0c;它简化了网页设计与开发流程&#xff0c;帮助开发者快速创建响应式布局和美观的网页界面。本文将向您介绍如何在项目中引入 Bootstrap 3.x 版本的基本步骤&#xff0c;包括 CSS 和 JavaScript 文件的引用&#xff0c;以及必要的…

Java跨Docker容器备份数据库数据

Java跨Docker容器备份数据库数据 前置背景思路整理编写备份脚本容器启动检验效果Java容器MySQL容器 Java代码执行备份 我的个人博客&#xff1a;Lichg&#xff0c;欢迎大家访问。 前置背景 在我们的开发部署场景中&#xff0c;通常多数使用Docker进行部署。当你的数据库和项目…

【吊打面试官系列】Java高并发篇 - 什么是乐观锁和悲观锁?

大家好&#xff0c;我是锋哥。今天分享关于 【什么是乐观锁和悲观锁?】面试题&#xff0c;希望对大家有帮助&#xff1b; 什么是乐观锁和悲观锁? 1、乐观锁&#xff1a; 就像它的名字一样&#xff0c;对于并发间操作产生的线程安全问题持乐观状态&#xff0c; 乐观锁认为竞争…

ICML 2024 多视角融合驱动的通用具身操作算法SAM-E,为机器人学习复杂操作给出了可行解决方案

当我们拿起一个机械手表时&#xff0c;从正面会看到表盘和指针&#xff0c;从侧面会看到表冠和表链&#xff0c;打开手表背面会看到复杂的齿轮和机芯。每个视角都提供了不同的信息&#xff0c;将这些信息综合起来才能理解操作对象的整体三维。 想让机器人在现实生活中学会执行…

【stm32】江科协听课笔记

[3-1] GPIO输出_哔哩哔哩_bilibili 5.GPIO输出 这里&#xff0c;寄存器就是一段特殊的存储器&#xff0c;内核可以通过APB2总线队寄存器进行读写&#xff0c;这样就可以完成输出/读取电平的功能。寄存器的每一位对应一个引脚&#xff0c;stm32是32位的&#xff0c;这里的寄存器…

【工具】珍藏免费宝藏工具,不好用你来捶我

前言 &#x1f34a;缘由 藏在我硬盘里的好东西&#xff0c;必须跟小伙伴们分享 东西很好&#xff0c;必须分享。良辰吉日&#xff0c;跟大家分享几款神仙级小工具&#xff0c;实际亲测&#xff0c;不好来锤。 正文 一.影刀 影刀可以使任何操作系统、桌面软件、Web程序的自动…

怎么搭建微信留言板功能

在信息爆炸的时代&#xff0c;微信已经成为了我们日常生活中不可或缺的一部分。它不仅仅是一个简单的聊天工具&#xff0c;更是一个充满无限可能的营销平台。今天&#xff0c;我要向大家介绍的是如何在你的微信平台上搭建一个独具特色的留言板功能&#xff0c;让用户能够自由发…

PyTorch的数据处理

&#x1f4a5;今天看一下 PyTorch数据通常的处理方法~ 一般我们会将dataset用来封装自己的数据集&#xff0c;dataloader用于读取数据 Dataset格式说明 &#x1f4ac;dataset定义了这个数据集的总长度&#xff0c;以及会返回哪些参数&#xff0c;模板&#xff1a; from tor…

51-54 Sora能制作动作大片还需要一段时间 | DrivingGaussian:周围动态自动驾驶场景的复合高斯飞溅

24年3月&#xff0c;北大、谷歌和加州大学共同发布了DrivingGaussian: Composite Gaussian Splatting for Surrounding Dynamic Autonomous Driving Scenes。视图合成和可控模拟可以生成自动驾驶的极端场景Corner Case&#xff0c;这些安全关键情况有助于以更低成本验证和增强自…