【golang项目-GeeCache】动手写分布式缓存 day1 - 实现LRU算法

news2024/11/24 13:37:24

介绍 LRU 内存淘汰算法

LRU(Least Recently Used) 最近最少使用 算法 ,系统认为如果这个数据最近使用过那么它被再次使用的概率会高,所以系统会先淘汰最久没被使用的数据

基本逻辑

在这里插入图片描述

-----------------------------------------------------------------------出自极客兔兔
k(绿色)为map,即实际中的缓存,当我们读取数据时就是先从这个查询和获取,复杂度为log 级别,非常快
n(红色)为双向链表,用来记录哪个数据是最晚出现的,使用双向链表的原因是为了快速让数据放在队首/队尾,当我们访问一个数据时,我们把这个数据放在链表队首,当我们需要淘汰内存时我们删除队尾的数据,这个数据就是最晚出现的

具体实现

实现数据结构

  1. 实现Cache,包含允许的最大缓存,当前的缓存,双向链表,map,回调函数
type Cache struct {
    maxBytes  int64                         // 允许的最大内存
    nbytes    int64                         // 已使用的内存
    ll        *list.List                    //双向链表
    cache     map[string]*list.Element      // map
    OnEvicted func(key string, value Value) //是某条记录被移除时的回调函数
}
  1. 实现双向链表的数据类型,便于删除双向链表时根据key删除map的值
type entry struct { // 代表双向链表的数据类型
    key   string
    value Value
}
  1. 为了存取数据的通用性,实现Len()获取元素个数
type Value interface {  
	Len() int  
}

初始化缓存函数

func New(maxBytes int64, onEvicted func(string, Value)) *Cache { // 创建
    return &Cache{
        maxBytes:  maxBytes,
        ll:        list.New(),
        cache:     make(map[string]*list.Element),
        OnEvicted: onEvicted,
    }
}

实现查找功能

这个函数首先在缓存(map)中查询是否存在,如果存在就返回这个值,并且把这个值放在双向链表的首部,也标记成最新出现

func (c *Cache) Get(key string) (value Value, ok bool) { // 查找
    if ele, ok := c.cache[key]; ok {
        c.ll.MoveToFront(ele)    // 移动到队首
        kv := ele.Value.(*entry) // 找到值
        return kv.value, true
    }
    return
}

在这里写的时候 我对这段代码有疑惑

 kv := ele.Value.(*entry) // 找到值

错误的认为ele.Value.(T) 是 Value中的一个成员变量,但实际又不是
实际上这是接口类型的类型转换 Value.(Type) 是吧接口类型Value转换成Type类型

补充知识 :Go语言接口和类型之间的转换
类型断言

类型断言用于将接口类型转换为指定类型,其语法为:
value.(type) 或者 value.(T)

类型转换

类型转换用于将一个接口类型的值转换为另一个接口类型,其语法为:
T(value)

实现内存淘汰功能 即删除

内存删除就是移除最近最少访问的节点,就是删除队尾,然后修改当前缓存大小,使用回调函数通知系统

func (c *Cache) RemoveOldest() { // 缓存淘汰
    ele := c.ll.Back() // 取到队首节点,从链表中删除
    if ele != nil {    // 非空
        c.ll.Remove(ele) // 移除最近最少访问
        kv := ele.Value.(*entry)
        delete(c.cache, kv.key)
        c.nbytes -= int64(len(kv.key)) + int64(kv.value.Len())
        if c.OnEvicted != nil {
            c.OnEvicted(kv.key, kv.value) // 回调
        }
    }

}

实现添加/修改数据操作

添加/修改数据时查询缓存是否存在,否则在缓存中(map)中创建个新的键值对,最后把这个数据放在队首,表示最新出现

func (c *Cache) Add(key string, value Value) {
    if ele, ok := c.cache[key]; ok {
        c.ll.MoveToFront(ele)
        kv := ele.Value.(*entry)
        c.nbytes += int64(value.Len()) - int64(kv.value.Len())
        kv.value = value
    } else {
        ele := c.ll.PushFront(&entry{key, value})
        c.cache[key] = ele
        c.nbytes += int64(len(key)) + int64(value.Len())
    }
    for c.maxBytes != 0 && c.maxBytes < c.nbytes {
        c.RemoveOldest() //如果超过了设定的最大值 c.maxBytes,则移除最少访问的节点。

    }

}

全部代码

实现代码

package lru

  

import "container/list"

  

type Cache struct {

    maxBytes  int64                         // 允许的最大内存

    nbytes    int64                         // 已使用的内存

    ll        *list.List                    //双向链表

    cache     map[string]*list.Element      // map

    OnEvicted func(key string, value Value) //是某条记录被移除时的回调函数

}

  

type entry struct { // 代表双向链表的数据类型

    key   string

    value Value

}

  

type Value interface {

    Len() int

}

  

func New(maxBytes int64, onEvicted func(string, Value)) *Cache { // 创建

    return &Cache{

        maxBytes:  maxBytes,

        ll:        list.New(),

        cache:     make(map[string]*list.Element),

        OnEvicted: onEvicted,

    }

}

  

func (c *Cache) Get(key string) (value Value, ok bool) { // 查找

    if ele, ok := c.cache[key]; ok {

        c.ll.MoveToFront(ele)    // 移动到队尾

        kv := ele.Value.(*entry) // 找到值

        return kv.value, true

    }

    return

}

  

func (c *Cache) RemoveOldest() { // 缓存淘汰

    ele := c.ll.Back() // 取到队首节点,从链表中删除

    if ele != nil {    // 非空

        c.ll.Remove(ele) // 移除最近最少访问

        kv := ele.Value.(*entry)

        delete(c.cache, kv.key)

        c.nbytes -= int64(len(kv.key)) + int64(kv.value.Len())

        if c.OnEvicted != nil {

            c.OnEvicted(kv.key, kv.value) // 回调

        }

    }

}

  

func (c *Cache) Add(key string, value Value) {

    if ele, ok := c.cache[key]; ok {

        c.ll.MoveToFront(ele)

        kv := ele.Value.(*entry)

        c.nbytes += int64(value.Len()) - int64(kv.value.Len())

        kv.value = value

    } else {

        ele := c.ll.PushFront(&entry{key, value})

        c.cache[key] = ele

        c.nbytes += int64(len(key)) + int64(value.Len())

    }

    for c.maxBytes != 0 && c.maxBytes < c.nbytes {

        c.RemoveOldest() //如果超过了设定的最大值 c.maxBytes,则移除最少访问的节点。

    }

}

  

func (c *Cache) Len() int {

    return c.ll.Len()

}

测试代码
在这里也可以学会testing库的使用

package lru

  

import (

    "reflect"

    "testing"

)

  

type String string

  

func (d String) Len() int {

    return len(d)

}

  

func TestGet(t *testing.T) {

    lru := New(int64(0), nil)

    lru.Add("key1", String("1234"))

    if v, ok := lru.Get("key1"); !ok || string(v.(String)) != "1234" {

        t.Fatalf("cache hit key1=1234 failed")

    }

    if _, ok := lru.Get("key2"); ok {

        t.Fatalf("cache miss key2 failed")

    }

}

  

func TestRemoveoldest(t *testing.T) {

    k1, k2, k3 := "key1", "key2", "k3"

    v1, v2, v3 := "value1", "value2", "v3"

    cap := len(k1 + k2 + v1 + v2)

    lru := New(int64(cap), nil)

    lru.Add(k1, String(v1))

    lru.Add(k2, String(v2))

    lru.Add(k3, String(v3))

  

    if _, ok := lru.Get("key1"); ok || lru.Len() != 2 {

        t.Fatalf("Removeoldest key1 failed")

    }

}

  

func TestOnEvicted(t *testing.T) {

    keys := make([]string, 0)

    callback := func(key string, value Value) {

        keys = append(keys, key)

    }

    lru := New(int64(10), callback)

    lru.Add("key1", String("123456"))

    lru.Add("k2", String("k2"))

    lru.Add("k3", String("k3"))

    lru.Add("k4", String("k4"))

  

    expect := []string{"key1", "k2"}

  

    if !reflect.DeepEqual(expect, keys) {

        t.Fatalf("Call OnEvicted failed, expect keys equals to %s", expect)

    }

}

  

func TestAdd(t *testing.T) {

    lru := New(int64(0), nil)

    lru.Add("key", String("1"))

    lru.Add("key", String("111"))

  

    if lru.nbytes != int64(len("key")+len("111")) {

        t.Fatal("expected 6 but got", lru.nbytes)

    }

}

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

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

相关文章

手把手教你学习IEC104协议和编程实现 十一-定值的概念讲解、定值的操作过程以及部分代码的实现

从本章开始,我们开始研究定值部分; 定值是什么? 了解过终端的可能都知道,定值就是保护定值,就是设定了一组参数,当终端的采样值达到这个参数的时候,终端就会做出一系列的反应。这样的目的,是为了保护电网,让电网正常运行,具体为什么这么做,不做详细的解释,如果有…

李宏毅2021春季机器学习课程视频笔记13-自注意力机制

【(强推)李宏毅2021/2022春机器学习课程】 Slide地址 一、问题引入 1.模型的输入 无论是预测视频观看人数、视频处理、语言识别&#xff0c;这些所有的model中&#xff0c;输入数据都可以视作为一个向量&#xff08;vector&#xff09;&#xff0c;模型的输出为一个数值或者一…

UDP的报文结构及注意事项

UDP的报文结构及注意事项&#x1f50e;UDP的报文结构源端口和目的端口报文长度校验和&#x1f50e;UDP的注意事项端口号报文长度校验和&#x1f50e;结尾&#x1f50e;UDP的报文结构 图片来自网络 源端口和目的端口 如果将 源IP 和 目的IP 看作是两台计算机在网络中的地址 那么…

完美解决丨#在python中,如果引用的变量未定义,则会报告NameError: name ‘变量名‘ is not defined。

NameError 在python中&#xff0c;如果引用的变量未定义&#xff0c;则会报告NameError: name 变量名 is not defined。 如下代码抛出了一个异常&#xff1a; !/usr/bin/env python -- coding:utf-8 -- print hello world print hello %s % name 报错信息如下&#xff1a; Trac…

基于springboot和ajax的简单项目 02 代码部分实现,思路 (上)

01.由于是对功能的实现&#xff0c;应该是按照功能的需要去写代码&#xff0c;所以&#xff0c;先看前端html文件的代码。 02.项目的开始界面是starter.html文件。 关键的script标签 <script type"text/javascript">$(function(){//页面加载完成之后执行doLo…

VS中解决方案和项目的区别

总目录 文章目录总目录一、概述1、解决方案2、项目3、项目文件4、解决方案文件夹二、图解1、图解解决方案和项目的关系2、图解sln文件3、图解项目文件结语一、概述 1、解决方案 解决方案是一个容器&#xff0c;通常包含多个项目&#xff0c;这些项目通常相互引用。 解决方案中…

CSDN粉丝首破一千关,有你名字

2023-4-11&#xff0c;CSDN粉丝首破一千关。 感谢词版本1,哈哈哈哈哈哈哈哈 在编程世界里&#xff0c;人们可以像创造生命一样创造程序&#xff0c;而我对这种创造和创新的热情&#xff0c;从我的csdn博客社区粉丝首次突破一千人的消息中得到了极大的满足和激励。作为一个Pyth…

Spring中Bean初始化和销毁的多种方式

Spring中Bean初始化和销毁的多种方式一、Bean的多种初始化方式1.PostConstruct注解2.实现InitializingBean接口3.声明init-method方法二、Bean的多种销毁方式1.PreDestroy注解2.实现DisposableBean接口3.声明destroy-method方法三、总结Spring中支持在Bean的加载时声明初始化方…

跑得快的不止是程序丄【掌握自动化测试让你过五关斩六将】

拥有自动化测试技能的软件测试人员更具竞争力&#xff0c;这是当下面试过的人都非常认同的一句话。 作为一名软件测试人员&#xff0c;我们都知道“时间就是金钱”&#xff0c;尤其是在快速迭代的敏捷开发模式下&#xff0c;更是如此。在传统的软件测试流程中&#xff0c;手工…

【MySQL】JDBC编程

摄影分享 目录 数据库编程的必备条件 Java的数据库编程&#xff1a;JDBC JDBC的使用步骤 1. 创建数据源DataSourece 2.连接数据库 3.构造并执行sql语句 4.遍历结果集合 5.释放资源 数据库编程的必备条件 编程语言&#xff0c;如Java&#xff0c;C、C、Python等数据库&am…

谁说35岁是程序员的中年危机?那是他还不知道这些新路子

文章目录一、年纪大能不能进大厂&#xff1f;二、为什么说35是危机&#xff1f;1.精力衰退2.脑力衰退3.知识/技术迭代三、年龄大的程序员有哪些出路&#xff1f;1.技术管理2.创业3.技术外包4.做老师5.做自媒体6.写书四、结语我自己今年已有44了&#xff0c;从2021年开始就已经不…

详解自动化测试之 Selenium 与 Junit

文章目录1. 什么是自动化2. 自动化测试的分类3. selenium&#xff08;web 自动化测试工具&#xff09;4. 一个简单的自动化例子5. selenium 常用方法5.1 查找页面元素 findElement ()5.2 元素的定位 By 类5.3 xpath 路径语言6. 常见的元素操作6.1 输入文本 sendKeys6.2 点击 cl…

什么是Android FrameWork,请你介绍一下?

Framework是什么 Framework的中文意思是“框架”&#xff0c;在软件开发中通常指开发框架&#xff0c;在一个系统中处于内核层之上&#xff0c;为顶层应用提供接口&#xff0c;被设计用来帮助开发者快速开发顶层应用&#xff0c;而不必关心系统内核运行机制&#xff0c;通常Fr…

zabbix报警配置

一、前言 这里用的是zabbix6.0LTS版本&#xff0c;这里记录自定义配置报警&#xff0c;因为邮件报警基本已经很少有人使用了&#xff0c;大部分是&#xff0c;短信、飞书、钉钉等等工具&#xff0c;所有需要定制化报警 二、定义脚本存放路径 cd /usr/local/zabbix/etc[rootn…

MODBUS 转 EtherNet/IP 网关连接希望森兰变频器案例

Modbus转Ethernet/Ip网关&#xff0c;用于将多个 MODBUS 从站设备接入 ETHERNET/IP 主站网络&#xff0c;实现 MODBUS 转 ETHERNET/IP 功能。配上 MODBUS 转 EtherNet 网关专用的 EDS 文件,实现 ETHERNET/IP 主 站对 MODBUS 从站设备的控制。 需要设备 .24v电源模块 罗克韦尔PL…

【城市污水处理过程中典型异常工况智能识别】(基于迁移学习,拓扑结构卷积神经网络的污水异常工况识别)

基于迁移学习拓扑结构卷积神经网络的污水异常工况识别 **摘 要&#xff1a;针对城市污水处理过程的异常工况识别问题&#xff0c;本文提出了基于图像纹理性分析的工况识别方法。首先总结了几种典型的异常工况的特点&#xff0c;并且分析了卷积神经网络特征提取异常工况的几种纹…

月薪20k的性能测试必备技能:发现性能瓶颈掌握性能调优

背景 当下云计算、大数据盛行的背景下&#xff0c;大并发和大吞吐量的需求已经是摆在企业面前的问题了&#xff0c;其中网络的性能要求尤为关键&#xff0c;除了软件本身需要考虑到性能方面的要求&#xff0c;一些硬件上面的优化也是必不可少的。 作为一名测试工作者&#xf…

ubuntu将主文件夹的文件夹中文名称改为英文

许多人在使用ubuntu时使用汉语的&#xff0c;但是主文件夹下的文件夹也随之变成了中文&#xff0c;导致命令行下输入文件夹名称非常的困难&#xff0c;那么怎么简单的把名字改成英文的呢&#xff1f;&#xff08;也不知道哪个版本开始不出现下面这个图示提示了&#xff09; 方法…

Vue2-黑马(十二)

目录&#xff1a; &#xff08;1&#xff09;vue2-登录-store-user.js &#xff08;2&#xff09;实战-登录-store-user.js &#xff08;3&#xff09;登录-实战--permission.js-获取角色 &#xff08;1&#xff09;vue2-登录-store-user.js 我们进入store中的actions查看它…

MySQL - 基于SSL安全连接的主从复制

目录 &#x1f341;主从复制的原理 &#x1f341;部署master &#x1f341;部署slave &#x1f341;测试SSL主从复制 &#x1f990;博客主页&#xff1a;大虾好吃吗的博客 &#x1f990;MySQL专栏&#xff1a;MySQL专栏地址 生产环境中一台mysql主机存在单点故障&#xff0c;所…