使用go语言的list实现一个简单的LRU缓存

package main;

import (
	"container/list"
	"errors"
	"sync"
	"fmt"
	"encoding/json"
)

//LRU(Least recently used)最近最少使用,算法根据数据的历史访问记录来进行淘汰数据
//核心思想是"如果数据最近被访问过,那么将来被访问的几率也更高"
//常见的实现方式是用一个链表保存数据
//1. 新数据插入到链表头部
//2. 每当缓存命中(即缓存数据被访问),则将数据移到链表头部
//3. 当链表满的时候,将链表尾部的数据丢弃

type cacheItem struct {
	Key string;
	Val interface{};
}

type LRU struct {
	//最大存储数量
	maxNum int;
	//当前存储数量
	curNum int;
	//锁,保证数据一致性
	mutex  sync.Mutex;
	//链表
	data   *list.List;
}

//添加数据
func (l *LRU) add(key string,value interface{}) error {
	//判断key是否存在
	if e,_ := l.exist(key); e {
		return errors.New(key + "已存在");
	}
	//判断当前存储数量与最大存储数量
	if l.maxNum == l.curNum {
		//链表已满,则删除链表尾部元素
		l.clear();
	}
	l.mutex.Lock();
	l.curNum++;
	//json序列化数据
	data,_ := json.Marshal(cacheItem{key,value});
	//把数据保存到链表头部
	l.data.PushFront(data);
	l.mutex.Unlock();
	return nil;
}

//设置数据
func (l *LRU) set(key string,value interface{}) error {
	e,item := l.exist(key);
	if !e {
		return l.add(key,value);
	}
	l.mutex.Lock();
	data,value});
	//设置链表元素数据
	item.Value = data;
	l.mutex.Unlock();
	return nil;
}

//清理数据
func (l *LRU) clear() interface{} {
	l.mutex.Lock();
	l.curNum--;
	//删除链表尾部元素
	v := l.data.Remove(l.data.Back());
	l.mutex.Unlock();
	return v;
}

//获取数据
func (l *LRU) get(key string) interface{} {
	e,item := l.exist(key);
	if !e {
		return nil;
	}
	l.mutex.Lock();
	//数据被访问,则把元素移动到链表头部
	l.data.MoveToFront(item);
	l.mutex.Unlock();
	var data cacheItem;
	json.Unmarshal(item.Value.([]byte),&data);
	return data.Val;
}

//删除数据
func (l *LRU) del(key string) error {
	e,item := l.exist(key);
	if !e {
		return errors.New(key + "不存在");
	}
	l.mutex.Lock();
	l.curNum--;
	//删除链表元素
	l.data.Remove(item);
	l.mutex.Unlock();
	return nil;
}

//判断是否存在
func (l *LRU) exist(key string) (bool,*list.Element) {
	var data cacheItem;
	//循环链表,判断key是否存在
	for v := l.data.Front(); v != nil; v = v.Next() {
		json.Unmarshal(v.Value.([]byte),&data);
		if key == data.Key {
			return true,v;
		}
	}
	return false,nil;
}

//返回长度
func (l *LRU) len() int {
	return l.curNum;
}

//打印链表
func (l *LRU) print() {
	var data cacheItem;
	for v := l.data.Front(); v != nil; v = v.Next() {
		json.Unmarshal(v.Value.([]byte),&data);
		fmt.Println("key:",data.Key," value:",data.Val);
	}
}

//创建一个新的LRU
func LRUNew(maxNum int) *LRU {
	return &LRU{
		maxNum: maxNum,curNum: 0,mutex:  sync.Mutex{},data:   list.New(),};
}

func main() {
	lru := LRUNew(5);
	lru.add("1111",1111);
	lru.add("2222",2222);
	lru.add("3333",3333);
	lru.add("4444",4444);
	lru.add("5555",5555);
	lru.print();
	//get成功后,可以看到3333元素移动到了链表头
	fmt.Println(lru.get("3333"));
	lru.print();
	//再次添加元素,如果超过最大数量,则删除链表尾部元素,将新元素添加到链表头
	lru.add("6666",6666);
	lru.print();
	lru.del("4444");
	lru.print();
	lru.set("2222","242424");
	lru.print();
}

  

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


Golang的文档和社区资源:为什么它可以帮助开发人员快速上手?
Golang:AI 开发者的实用工具
Golang的标准库:为什么它可以大幅度提高开发效率?
Golang的部署和运维:如何将应用程序部署到生产环境中?
高性能AI开发:Golang的优势所在
本篇文章和大家了解一下go语言开发优雅得关闭协程的方法。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。1.简介本文将介绍首先为什么需要主...
这篇文章主要介绍了Go关闭goroutine协程的方法,具有一定借鉴价值,需要的朋友可以参考下。下面就和我一起来看看吧。1.简介本文将介绍首先为什么需要主动关闭gor...
本篇文章和大家了解一下go关闭GracefulShutdown服务的几种方法。有一定的参考价值,有需要的朋友可以参考一下,希望对大家有所帮助。目录Shutdown方法Regi...
这篇文章主要介绍了Go语言如何实现LRU算法的核心思想和实现过程,具有一定借鉴价值,需要的朋友可以参考下。下面就和我一起来看看吧。GO实现Redis的LRU例子常
今天小编给大家分享的是Go简单实现多租户数据库隔离的方法,相信很多人都不太了解,为了让大家更加了解,所以给大家总结了以下内容,一起往下看吧。一定会...
这篇“Linux系统中怎么安装NSQ的Go语言客户端”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希
本文小编为大家详细介绍“怎么在Go语言中实现锁机制”,内容详细,步骤清晰,细节处理妥当,希望这篇“怎么在Go语言中实现锁机制”文章能帮助大家解决疑惑,下面...
今天小编给大家分享一下Go语言中interface类型怎么使用的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考
这篇文章主要介绍“怎么以正确的方式替换Go语言程序自身”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希
本文小编为大家详细介绍“Go语言中除法运算的效率怎么提高”,内容详细,步骤清晰,细节处理妥当,希望这篇“Go语言中除法运算的效率怎么提高”文章能帮助大家解...
本文小编为大家详细介绍“Go语言中的next()方法怎么使用”,内容详细,步骤清晰,细节处理妥当,希望这篇“Go语言中的next()方法怎么使用”文章能帮助大家解决疑...
这篇文章主要介绍了Go语言中slice的反转方法怎么使用的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇Go语言中slice的反转方法怎...
这篇文章主要介绍“怎么使用Go语言实现数据转发功能”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“怎么使用Go语
这篇文章主要讲解了“Go语言中怎么实现代码跳转”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究
这篇文章主要讲解了“Go语言如何多开协程”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Go语言如何多开协...