LFU算法 初始频率 动态频率
LFU(Least Frequently Used)算法是一种缓存淘汰策略,其核心思想是根据数据的访问频率来决定淘汰哪些数据。具体来说,
LFU算法认为如果一个数据在过去一段时间内被访问的次数很少,那么它在未来被再次访问的概率也很低。因此,当缓存空间不足时,LFU算法会选择访问频率最低的数据进行淘汰。
在C++中实现LFU算法,通常需要以下几个步骤:
数据结构设计:LFU算法通常需要一个哈希表和一个优先队列。哈希表用于存储每个元素的访问计数,键是元素的标识,值是元素的访问次数。优先队列用于根据访问次数对元素进行排序,以便快速找到访问次数最少的元素。
初始化:在初始化时,需要设置缓存的容量,并创建一个哈希表和一个优先队列来存储数据和访问计数。
获取数据:当调用get方法时,如果键存在于缓存中,则返回键的值,并增加该键的访问计数。如果键不存在,则返回-1。
插入数据:当调用put方法时,如果键已存在,则更新其值并增加访问计数;如果键不存在,则插入新键值对,并检查缓存是否已满。如果已满,则淘汰访问次数最少的元素。
淘汰策略:在淘汰元素时,LFU算法会选择访问次数最少的元素。如果有多个元素具有相同的最小访问次数,则选择最早插入的那个元素进行淘汰。
1 基于初始的频率
- LRUCache(int capacity) 以正整数作为容量 capacity 初始化 LRU 缓存。
- int get(int key) 如果关键字 key 存在于缓存中,则返回关键字的值,否则返回 - 1。
- void put(int key, int value) 如果关键字已经存在,则变更其数据值;如果关键字不存在,则插入该组「关键字 - 值」。
- 当缓存容量达到上限时,它应该在写入新数据之前删除最久未使用的数据值,从而为新的数据值留出空间。
- 在 O(1) 时间复杂度内完成这两种操作。
C++写的伪代码如下:
class LFUCache {
private:int m_nCapacity; //缓存的容量unordered_map<int, list<pair<int, int>>::iterator> m_Iter; //链表索引list<pair<int, int>> m_List; //(key,value) 链表public:LFUCache(int capacity){m_nCapacity = capacity;m_Iter.clear();m_List.clear();}int Get(int key){unordered_map<int, list<pair<int, int>>::iterator>::iterator iter = m_Iter.find(key);if (iter == m_Iter.end())return -1;UpdateNode(key, m_Iter[key]->second);pair<int,int> tail = m_List.back();return tail.second;}void Put(int key, int value){if (m_Iter.find(key) != m_Iter.end()){UpdateNode(key, value);return;}//若容量已满,则移除"最近最少使用的节点"if (m_List.size() >= m_nCapacity){DelLeastNode(key);}AddNode(key,value);}private:void UpdateNode(int key, int value){m_List.erase(m_Iter[key]); //删除原有的pairAddNode(key, value);}void AddNode(int key, int value){m_List.push_back(make_pair(key, value));list<pair<int, int>>::iterator iter = m_List.end();m_Iter[key] = --iter;}void DelLeastNode(int key){int id = m_List.begin()->first;m_List.erase(m_List.begin());m_Iter.erase(id);}};
2 基于动态的频率
每读一次,该项的热度加1;每写一次,该项的热点也加1;
-
LFUCache(int capacity) - 用数据结构的容量 capacity 初始化对象。
-
int get(int key) - 如果键存在于缓存中,则获取键的值,否则返回 - 1。
-
void put(int key, int value) - 如果键已存在,则变更其值;如果键不存在,请插入键值对。当缓存达到其容量时,
-
则应该在插入新项之前,使最不经常使用的项无效。在此问题中,当存在平局(即两个或更多个键具有相同使用频率)时,
-
应该去除 最近最久未使用的键。
-
「项的使用次数」就是自插入该项以来对其调用 get 和 put 函数的次数之和。使用次数会在对应项被移除后置为 0 。
-
为了确定最不常使用的键,可以为缓存中的每个键维护一个 使用计数器 。使用计数最小的键是最久未使用的键。
-
当一个键首次插入到缓存中时,它的使用计数器被设置为 1 (由于 put 操作)。对缓存中的键执行 get 或 put 操作,使用计数器的值将会递增。
C++写的伪代码如下:
class LFUCache {
private:int m_nCapacity; //容量int m_nMinFreq; //最小的使用次数unordered_map<int, pair<int, int>> m_map; //(key,value,freq) 三元组unordered_map<int, list<int>> m_freqList; //(freq,key) 频次链表unordered_map<int, list<int>::iterator> m_lisIter; //(freq,index) 频次索引器LFUCache(int capacity){m_nCapacity = capacity;m_nMinFreq = 0;m_map.clear();m_freqList.clear();m_lisIter.clear();}void IncreaseFreq(int key){int oldFreq = m_map[key].second++;//在链表里删除旧元素m_freqList[oldFreq].erase(m_lisIter[key]);//添加元素,并更新次数m_freqList[oldFreq + 1].emplace_front(key); //从双端队列的头部加入元素m_lisIter[key] = m_freqList[oldFreq + 1].begin();if (m_freqList[m_nMinFreq].empty()){m_nMinFreq = oldFreq + 1;}}void AddNode(int key, int value){m_nMinFreq = 1;m_map[key] = make_pair(value, m_nMinFreq);m_freqList[m_nMinFreq].emplace_front(key);m_lisIter[key] = m_freqList[m_nMinFreq].begin();}int Get(int key){if (m_map.find(key) == m_map.end())return -1;IncreaseFreq(key);return m_map[key].first;}void Put(int key, int value){if (m_nCapacity <= 0)return;//若key存在,则更新value值if (m_map.find(key) != m_map.end()){m_map[key].first = value;IncreaseFreq(key);return;}//处理容量已满的情况if (m_map.size() >= m_nCapacity){int id = m_freqList[m_nMinFreq].back();m_freqList[m_nMinFreq].pop_back(); //弹出末尾的元素m_lisIter.erase(id); //删除该元素的索引m_map.erase(id); //删除该元素}//若key不存在,则新建AddNode(key, value);}};
相关文章:
LFU算法 初始频率 动态频率
LFU(Least Frequently Used)算法是一种缓存淘汰策略,其核心思想是根据数据的访问频率来决定淘汰哪些数据。具体来说, LFU算法认为如果一个数据在过去一段时间内被访问的次数很少,那么它在未来被再次访问的概率也…...
Spring Boot 进阶-详解SpringBoot的复杂数据校验规则
在之前的文章中,我们介绍了SpringBoot整合JSR-303规则来完成数据校验操作。接下来我们来聊一聊关于数据校验的具体用法。 之前的文章中举过一个简单的例子通过学生信息提交的例子来介绍了关于数据校验如何去做。那么接下来这篇文章,我们就来看看对于一些复杂的数据校验如何完…...
wsl环境下安装Ubuntu,并下载MySQL5.7
安装操作需root权限,切换root用户有两种方式: 1-通过 sudo su - ,切换到root用户(登录后长期有效)。 2-在每一个命令前加上sudo,临时提升权限(仅对一条命令有效)。 1、下载apt仓库…...
倪师学习笔记-天纪-01
一、概要 介绍课程内容,介绍部分概念 二、具体内容 1、天纪内容 天机道:看象,使用斗数等工具人间道:看卦,使用易经地脉道:看风水地理 2、神 神与形对应,形是神的实例,神是形的…...
深入理解缓存穿透、缓存击穿和缓存雪崩
在现代分布式系统中,缓存是提升系统性能和减轻数据库负载的重要组件。然而,在实际应用中,我们可能会遇到一些缓存问题,如缓存穿透、缓存击穿和缓存雪崩。本文将详细探讨这三种缓存问题的原理、影响以及解决方案。 一,…...
【玩转动态规划专题】70. 爬楼梯【简单】
【玩转动态规划专题】70. 爬楼梯【简单】 1、力扣链接 https://leetcode.cn/problems/climbing-stairs/description/ 2、题目描述 假设你正在爬楼梯。需要 n 阶你才能到达楼顶。 每次你可以爬 1 或 2 个台阶。你有多少种不同的方法可以爬到楼顶呢? 示例 1&…...
前端开发设计模式——组合模式
目录 一、组合模式的定义和特点 1.定义 2.特点: 二、组合模式的实现方式 1.定义抽象组件类 2.创建叶节点类 3.创建组合类: 三、组合模式的应用场景 1.界面布局管理 2.菜单系统构建 3.组件库开发 四、组合模式的优点 1.简化客户端代码 2.增…...
初探OceanBase 4.x单机环境下如何进行主备架构搭建
本文来自OceanBase 用户的体验分享 (以下简称 OB),已经开源了3年左右,其间从3.x版本演进至4.x版本,发生了许多变化。对一个DBer而言,最为关切的是如何高效运用OB,以及是否能实现如同应用MySQL般…...
python 实现Edmonds-Karp算法
Edmonds-Karp算法介绍 Edmonds-Karp算法是一种用于解决最大流问题的算法,在计算机科学中广泛应用。以下是关于Edmonds-Karp算法的详细解释: 算法概述 Edmonds-Karp算法是基于Ford-Fulkerson方法的改进,它通过广度优先搜索(BFS&…...
【牛客刷题实战】BC120 争夺前五名
大家好,我是小卡皮巴拉 文章目录 目录 牛客题目: BC120 争夺前五名 题目描述 输入描述: 输出描述: 示例1 示例2 解题思路: 具体思路: 题目要点: 完整代码: 兄弟们共…...
WMS 智慧仓储管理系统的可视化管理_SunWMS
【大家好,我是唐Sun,唐Sun的唐,唐Sun的Sun。一站式数智工厂解决方案服务商】 WMS 智慧仓储管理系统的可视化管理主要表现在以下几个方面: 首先是库存可视化。通过系统,仓库管理人员能够以直观的图表、图形等形式清晰地…...
动态代理代码示例
理解动态代理 动态代理的核心在于代理对象的创建和方法调用是在运行时动态发生的,而不是在编译时就已经确定的性能监控、事务管理、日志记录通常需要使用代理对象对目标对象的功能进行增强为什么JDK动态代理只能代理有接口的类? 因为Proxy.newProxyIns…...
SpringBoot+Activiti7工作流使用进阶实例-高亮显示BPMN流程图( SpringBoot+Activiti+mybatis+shiro实现)
文章目录 说明绘制流程图排他网关设置任务节点设置创建工程修改 pom.xml 文件准备数据库的表和测试数据修改 application.yml 文件配置静态资源Shiro 相关配置ShiroConfiguration.javaMyShiroRealm.java流程控制器添加静态的资源和模板页面运行结果截图源码地址说明 使用 Spri…...
C#使用Lazy<T>提高性能
以下是一些适合使用Lazy<T>的场景: 单例模式 在实现单例模式时,Lazy<T>是非常有用的。如前面提到的示例,它可以确保单例对象在首次被访问时才进行创建,同时在多线程环境下也能保证正确的行为。这种方式比传统的双重检…...
创建读取比特币1P类型地址
创建读取比特币1P类型地址 比特币的地址类型有多种,其中 P2TR(Pay-to-Taproot)地址是基于最近的升级(Taproot)引入的一个新类型。本文将介绍如何创建和读取比特币的 1P 类型地址,主要通过 JavaScript 和相…...
从零开始Hadoop集群环境搭建
目录 1. Centos7.5硬件配置1.1 创建虚拟机1.2 虚拟机系统设置 2. IP地址和主机名称配置3. 软件配置3.1 安装 epel-release3.2 卸载虚拟机自带的JDK3.3 克隆虚拟机3.4 修改克隆虚拟机的IP3.5 JDK安装3.6 Hadoop安装 4. Hadoop目录结构 1. Centos7.5硬件配置 1.1 创建虚拟机 1.2…...
Copley耐环境伺服驱动器 极端环境下高精度控制解决方案
全球工业环境的日益复杂多变,对伺服驱动器的要求不再局限于基本的性能参数,而是在极端环境下的稳定性与可靠性。Copley耐环境伺服驱动器以卓越的性能和出色的环境适应性,为工业自动化领域的高精度控制提供了可靠的解决方案。 一、多样化的产…...
前端的全栈混合之路Meteor篇:分布式数据协议DDP深度剖析
本文属于进阶篇,并不是太适合新人阅读,但纯粹的学习还是可以的,因为后续会实现很多个ddp的版本用于web端、nodejs端、安卓端和ios端,提前预习和复习下。ddp协议是一个C/S架构的协议,但是客户端也同时可以是服务端。 什…...
基于Zynq SDIO WiFi移植一(支持2.4/5G)
基于SDIO接口的WIFI,在应用上,功耗低于USB接口,且无须USB Device支持,满足某些应用场景 1 硬件连接 2 Vivado工程配置 3 驱动编译 3.1 KERNRL CONFIG (build ENV) 修改 export KERNELPATH<path of kernel header>export T…...
数据结构与算法篇(刷题篇 - 链表)
目录 1. 反转链表(简单) 1.1. 题目描述 1.2. 解题思路 方法一:迭代(推荐使用) 方法二:递归(扩展思路) 方法三:使用栈解决 方法四:双链表求解 2. 链表内…...
SMUDebugTool终极指南:轻松解锁AMD Ryzen处理器的隐藏性能
SMUDebugTool终极指南:轻松解锁AMD Ryzen处理器的隐藏性能 【免费下载链接】SMUDebugTool A dedicated tool to help write/read various parameters of Ryzen-based systems, such as manual overclock, SMU, PCI, CPUID, MSR and Power Table. 项目地址: https:…...
圣邦微电子冲刺港股:年营收39亿,净利5.3亿 派息1亿 已获IPO备案
雷递网 雷建平 4月2日圣邦微电子(北京)股份有限公司(简称:“圣邦微电子”)日前更新招股书,准备在港交所上市。圣邦微电子已在A股上市,截至今日收盘,圣邦微电子股价为67.45元…...
彻底解决Windows磁盘空间危机:Driver Store Explorer专业驱动管理指南
彻底解决Windows磁盘空间危机:Driver Store Explorer专业驱动管理指南 【免费下载链接】DriverStoreExplorer Driver Store Explorer 项目地址: https://gitcode.com/gh_mirrors/dr/DriverStoreExplorer 你是否曾为Windows系统盘空间不断缩小而烦恼ÿ…...
ESTree节点遍历终极指南:深度优先与广度优先算法完整解析
ESTree节点遍历终极指南:深度优先与广度优先算法完整解析 【免费下载链接】estree The ESTree Spec 项目地址: https://gitcode.com/gh_mirrors/es/estree JavaScript开发者们,你们是否在构建代码分析工具时遇到过AST遍历的难题?&…...
毕业设计实战:基于SSM+MySQL的健身中心管理系统设计与实现全攻略
毕业设计实战:基于SSMMySQL的健身中心管理系统设计与实现全攻略 在开发“健身中心管理系统”毕业设计时,我曾因一个看似简单的场地预约与器材租赁的并发冲突问题,踩了一个“深坑”。初期设计时,仅简单地实现了场地预约和器材租赁的…...
OpenClaw版本升级:Qwen3-4B模型与新框架特性的兼容性
OpenClaw版本升级:Qwen3-4B模型与新框架特性的兼容性 1. 为什么需要关注版本升级 上周五晚上11点,我的OpenClaw突然弹出一条警告:"当前版本(v0.8.3)将在48小时后停止维护"。这个深夜警报让我意识到,是时候处理这个技术…...
Qwen3.5-2B轻量模型效果:20亿参数实现92%准确率的通用图文VQA任务
Qwen3.5-2B轻量模型效果:20亿参数实现92%准确率的通用图文VQA任务 1. 模型概述 Qwen3.5-2B是阿里云推出的轻量化多模态基础模型,属于Qwen3.5系列的小参数版本。这个仅20亿参数的模型在保持高性能的同时,显著降低了部署门槛和资源消耗。 核…...
Llama-3.2V-11B-cot实战教程:集成Whisper实现音视频+图像联合推理
Llama-3.2V-11B-cot实战教程:集成Whisper实现音视频图像联合推理 1. 项目概述与核心能力 Llama-3.2V-11B-cot是一个强大的视觉语言模型,它不仅能理解图像内容,还能进行系统性推理。这个模型基于LLaVA-CoT论文实现,特别适合需要结…...
会议纪要秒变问答库!WeKnora即时知识系统实战教程
会议纪要秒变问答库!WeKnora即时知识系统实战教程 1. 为什么你需要一个"不跑题"的会议助手? 想象这些常见的工作场景: 项目复盘会上,有人问"三个月前那次迭代的排期是怎样的?",所有…...
HP 现在可以零成本构建原生 iOS 和 Android 应用 NativePHP for Mobile v3 发布
插件化架构 v3 版本最大的变化是引入了模块化插件系统。此前版本中集成在核心包里的原生功能,现在被拆分成独立的插件。 每个插件都是一个独立的 Composer 包,包含 Swift 和 Kotlin 代码、权限清单以及原生依赖。开发者只需安装实际用到的插件…...
