面云账户时候问了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描述)的更多相关文章

  1. 再谈LRU双链表内存管理

    N年前我写了个双链表也发了博客,还添了代码.但是那个代码不但复杂,而且还有有问题的,一直懒得整理,放在空间误导别人.最近在写服务端,今天抽点空补一篇. 关于LRU网上随便搜,有过后端经验的人应该很多都 ...

  2. 双链表的基本实现与讲解(C++描述)

    双链表 双链表的意义 单链表相对于顺序表,确实在某些场景下解决了一些重要的问题,例如在需要插入或者删除大量元素的时候,它并不需要像顺序表一样移动很多元素,只需要修改指针的指向就可以了,其时间复杂度为 ...

  3. JAVA 链表操作:单链表和双链表

    主要讲述几点: 一.链表的简介 二.链表实现原理和必要性 三.单链表示例 四.双链表示例 一.链表的简介 链表是一种比较常用的数据结构,链表虽然保存比较复杂,但是在查询时候比较便捷,在多种计算机语言都 ...

  4. [C++11][数据结构]自己的双链表实现

    这个双链表,是我模仿stl的list制作的,只实现了一些基本功能,像merge,transfer这些就没有实现,用户可以用基本操作来自己做外部实现. 我没有选用stl的[begin,end)迭代器模式 ...

  5. 在单链表和双链表中删除倒数第K个节点

    [说明]: 本文是左程云老师所著的<程序员面试代码指南>第二章中“在单链表和双链表中删除倒数第K个节点”这一题目的C++复现. 本文只包含问题描述.C++代码的实现以及简单的思路,不包含解 ...

  6. Python与数据结构[0] -> 链表/LinkedList[1] -> 双链表与循环双链表的 Python 实现

    双链表 / Doubly Linked List 目录 双链表 循环双链表 1 双链表 双链表和单链表的不同之处在于,双链表需要多增加一个域(C语言),即在Python中需要多增加一个属性,用于存储指 ...

  7. java实现双链表(差点没写吐系列...)

    刚才把单链表写完了,现在又把双链表写了,双链表和单链表的区别就是每个节点有prior和next两个指针,不同于单链表的一个next指针,而且,正是因为有这两个指针,所以双链表可以前后两个方向去移动指针 ...

  8. 数据结构图文解析之:数组、单链表、双链表介绍及C++模板实现

    0. 数据结构图文解析系列 数据结构系列文章 数据结构图文解析之:数组.单链表.双链表介绍及C++模板实现 数据结构图文解析之:栈的简介及C++模板实现 数据结构图文解析之:队列详解与C++模板实现 ...

  9. C和指针 第十二章 使用结构和指针 双链表和语句提炼

    双链表中每个节点包含指向当前和之后节点的指针,插入节点到双链表中需要考虑四种情况: 1.插入到链表头部 2.插入到链表尾部 3.插入到空链表中 4.插入到链表内部 #include <stdio ...

  10. C#双链表

    单链表允许从一个结点直接访问它的后继结点,所以, 找直接后继结点的时间复杂度是 O(1).但是,要找某个结点的直接前驱结点,只能从表的头引用开始遍历各结点.如果某个结点的 Next 等于该结点,那么, ...

随机推荐

  1. 为什么会有unknown模块

    当一个模块在free后,他的线程仍然在运行时,会导致GetModuleName失败,返回unknown

  2. Games101 基于蒙特卡洛积分的光线路径追踪 作业7 框架解读

    目录 1 前言 2 辐射度量学 3 Coding Pseudo Code 3.1 uniform random point in triangle in 3D 3.2 Multithreading 3 ...

  3. [图像处理] 基于CleanVision库清洗图像数据集

    CleanVision是一个开源的Python库,旨在帮助用户自动检测图像数据集中可能影响机器学习项目的常见问题.该库被设计为计算机视觉项目的初步工具,以便在应用机器学习之前发现并解决数据集中的问题. ...

  4. JavaScript String 对象-常用知识点

    JavaScript String 对象-常用知识点 对象用于处理文本(字符串). 对象创建方法: new String(). String 对象属性 属性 描述 constructor 对创建该对象 ...

  5. CF#795 D - Max GEQ Sum O(n)做法

    D - Max GEQ Sum input 3 4 -1 1 -1 2 5 -1 2 -3 2 -1 3 2 3 -1 output YES YES NO 题目大意:给你一个数组a,让你任意一个字段均 ...

  6. MySQL中varchar(10)和varchar(100)的优缺点

    许多使用MySQL的同学都会使用到varchar这个数据类型.初学者刚开始学习varchar时,一定记得varchar是个变长的类型这个知识点,所以很多初学者在设计表时,就会把varchar(X)的长 ...

  7. optical simulation of quantum logic

    量子逻辑的光学模拟(PRA, 1998)  主机中<1998Cerf.pdf> 核心: 1. 用一个光子的多条路径的叠加态来表示n qubits, 那么实验上干涉仪所包含的路径数为 2^n ...

  8. 100 款支持 .NET 多版本的强大 WPF 控件库

    前言 推荐一款集成了超过100款控件的流行 XAML 控件库,同时提供了一系列常用的 .NET 帮助类-CookPopularUI.它可以简化开发流程,让我们能够更加专注于核心业务逻辑的实现. 让我们 ...

  9. 腾讯云 AI 代码助手:产品研发过程的思考和方法论

    文章摘要 本文将详细阐述 腾讯云 AI 代码助手的历史发展形态与产品整体架构,并从技术.研发方法论的角度分别阐述了产品的研发过程. 全文阅读约 5-8 分钟. 产品布局 AI 代码助手产品经历了三个时 ...

  10. 一文详解:项目如何从Docker慢慢演变成了K8s部署

    今天,我们将深入探讨一个项目部署的演变过程.在这篇文章中,为了紧扣主题,我们将从 Docker 开始讲解,分析为什么一个传统的项目逐步演变成了今天流行的 Kubernetes(K8s)集群部署架构.我 ...