LRU的map+双链表实现(Go描述)
面云账户时候问了LRU,具体实现的方式是map+双链表。Set和Get的时间复杂度都是O(1)。完整写一遍复习一下, 仅作记录
/**
* @Author: lzw5399
* @Date: 2021/5/20 22:28
* @Desc: 基于map和双链表实现的LRU算法
*/
package main
import "sync"
func main() {
lru := NewLRUCache(3)
lru.Set(1, 233)
lru.Set(2, 666)
lru.Set(3, 777)
lru.Set(5, 888)
lru.Get(2)
}
// LRUCache
type LRUCache struct {
capacity int
cache map[int]*LinkedNode
head *LinkedNode
tail *LinkedNode
sync.RWMutex
}
type LinkedNode struct {
key, value int
prev, next *LinkedNode
}
func NewLRUCache(capacity int) *LRUCache {
return &LRUCache{
capacity: capacity,
cache: make(map[int]*LinkedNode, capacity),
head: nil,
tail: nil,
RWMutex: sync.RWMutex{},
}
}
// - key是否已存在
// - 已存在, 将该节点移动到链表头部
// - 未存在, 判断cap是否已满
// - 满
// - 移除链表尾的节点
// - 新的node放入链表头
// - 新的node放入cache的map中
// - 未满
// - 新的node放入链表头
// - 新的node放入cache的map中
func (l *LRUCache) Set(key int, value int) {
l.RLock()
node, exist := l.cache[key]
l.RUnlock()
if exist {
l.moveToHead(node)
return
}
node = &LinkedNode{
key: key,
value: value,
}
l.Lock()
defer l.Unlock()
if l.capacity == len(l.cache) {
removedNode := l.removeTail()
delete(l.cache, removedNode.key)
}
l.addToHead(node)
l.cache[key] = node
}
// - 从map中获取是否存在
// - 不存在
// - 返回-1
// - 存在
// - 移到链表头部
// - 并返回具体的值
func (l *LRUCache) Get(key int) int {
l.RLock()
node, exist := l.cache[key]
l.RUnlock()
if !exist {
return -1
}
l.moveToHead(node)
return node.value
}
func (l *LRUCache) moveToHead(node *LinkedNode) {
l.removeNode(node)
l.addToHead(node)
}
func (l *LRUCache) removeTail() *LinkedNode {
return l.removeNode(l.tail)
}
func (l *LRUCache) removeNode(node *LinkedNode) *LinkedNode {
// 头节点
if node.prev == nil {
l.head = node.next
node.next.prev = nil
return node
}
// 尾节点
if node.next == nil {
l.tail = node.prev
node.prev.next = nil
return node
}
// 中间节点
node.prev.next = node.next
node.next.prev = node.prev
return node
}
func (l *LRUCache) addToHead(node *LinkedNode) {
if l.head == nil {
l.tail = node
} else {
l.head.prev = node
}
node.prev = nil
node.next = l.head
l.head = node
}
LRU的map+双链表实现(Go描述)的更多相关文章
- 再谈LRU双链表内存管理
N年前我写了个双链表也发了博客,还添了代码.但是那个代码不但复杂,而且还有有问题的,一直懒得整理,放在空间误导别人.最近在写服务端,今天抽点空补一篇. 关于LRU网上随便搜,有过后端经验的人应该很多都 ...
- 双链表的基本实现与讲解(C++描述)
双链表 双链表的意义 单链表相对于顺序表,确实在某些场景下解决了一些重要的问题,例如在需要插入或者删除大量元素的时候,它并不需要像顺序表一样移动很多元素,只需要修改指针的指向就可以了,其时间复杂度为 ...
- JAVA 链表操作:单链表和双链表
主要讲述几点: 一.链表的简介 二.链表实现原理和必要性 三.单链表示例 四.双链表示例 一.链表的简介 链表是一种比较常用的数据结构,链表虽然保存比较复杂,但是在查询时候比较便捷,在多种计算机语言都 ...
- [C++11][数据结构]自己的双链表实现
这个双链表,是我模仿stl的list制作的,只实现了一些基本功能,像merge,transfer这些就没有实现,用户可以用基本操作来自己做外部实现. 我没有选用stl的[begin,end)迭代器模式 ...
- 在单链表和双链表中删除倒数第K个节点
[说明]: 本文是左程云老师所著的<程序员面试代码指南>第二章中“在单链表和双链表中删除倒数第K个节点”这一题目的C++复现. 本文只包含问题描述.C++代码的实现以及简单的思路,不包含解 ...
- Python与数据结构[0] -> 链表/LinkedList[1] -> 双链表与循环双链表的 Python 实现
双链表 / Doubly Linked List 目录 双链表 循环双链表 1 双链表 双链表和单链表的不同之处在于,双链表需要多增加一个域(C语言),即在Python中需要多增加一个属性,用于存储指 ...
- java实现双链表(差点没写吐系列...)
刚才把单链表写完了,现在又把双链表写了,双链表和单链表的区别就是每个节点有prior和next两个指针,不同于单链表的一个next指针,而且,正是因为有这两个指针,所以双链表可以前后两个方向去移动指针 ...
- 数据结构图文解析之:数组、单链表、双链表介绍及C++模板实现
0. 数据结构图文解析系列 数据结构系列文章 数据结构图文解析之:数组.单链表.双链表介绍及C++模板实现 数据结构图文解析之:栈的简介及C++模板实现 数据结构图文解析之:队列详解与C++模板实现 ...
- C和指针 第十二章 使用结构和指针 双链表和语句提炼
双链表中每个节点包含指向当前和之后节点的指针,插入节点到双链表中需要考虑四种情况: 1.插入到链表头部 2.插入到链表尾部 3.插入到空链表中 4.插入到链表内部 #include <stdio ...
- C#双链表
单链表允许从一个结点直接访问它的后继结点,所以, 找直接后继结点的时间复杂度是 O(1).但是,要找某个结点的直接前驱结点,只能从表的头引用开始遍历各结点.如果某个结点的 Next 等于该结点,那么, ...
随机推荐
- Spire.Pdf打印PDF文件
1 /// <summary> 2 /// Spire.Pdf打印PDF文件 3 /// </summary> 4 /// <param name="fileN ...
- git常见问题Q&A
git基本命令解释 git restore --staged .:移除暂存区文件,不影响本地(撤销git add . 操作) git add -u:将删除文件的操作同步到暂存区.将本地的删除同步到版本 ...
- Redis学习笔记整理
一.Redis概述 1.redis简介 Redis(REmote DIctionary Server 远程字典服务器)是一款开源的,用ANSI C编写.支持网络.基于内存.亦可持久化的日志型.Key- ...
- 灵活轻巧的java接口自动化测试实战
前言 无论是自动化测试还是自动化部署,撸码肯定少不了,所以下面的基于java语言的接口自动化测试,要想在业务上实现接口自动化,前提是要有一定的java基础. 如果没有java基础,也没关系.这里小编也 ...
- 🥳重磅更新!Fluent Editor 开源富文本支持 LaTeX 可编辑公式啦~
你好,我是 Kagol,个人公众号:前端开源星球. Fluent Editor 是一个基于 Quill 2.0 的富文本编辑器,在 Quill 基础上扩展了丰富的模块和格式,框架无关.功能强大.开箱即 ...
- PHP数据库连接教程 - QSZ
1准备工作 首先,确保你的环境中已安装: PHP 7.0+ MySQL/MariaDB Web服务器(Apache/Nginx) 2数据库连接代码 // config.php setAttribute ...
- Windows 杜比OEM授权
我们中高端的windows笔记本上都可以看到Dolby音效,TV电视上也有支持Dolby显示选项. 杜比主要有几类:Dolby全景声(也叫Atmos).Dolby视界(Vision).杜比影院(Dol ...
- app&小程序&web安全—sign签名绕过
零.前言 在web界面登陆时,小程序和app传输数据时经常会碰到签名,会对请求的参数进行签名,如果自己修改了数据包就会校验失败非常麻烦. 本文编写的契机就是因为碰到了一个JeecgBoot的小程序, ...
- 一文搞懂flex(弹性盒布局)
1.什么是弹性布局 Flex是Flexible Box的缩写,翻译成中文就是"弹性盒子",用来为盒装模型提供最大的灵活性.任何一个容器都可以指定为Flex布局. 她是一种现代的CS ...
- Codeforces Round 892 (Div.2)
A. United We Stand 题解 赛时想复杂了 题目要求我们保证数组\(c\)中的数不是数组\(b\)中任意一个数的因子 我们考虑将最小值置于数组\(b\)即可 const int N = ...