Go语言LRU算法

简介: Go语言LRU算法
package main
import (
  "fmt"
)
// 定义节点结构体
type node struct {
  key   string
  value string
  next  *node
  prev  *node
}
// 定义lru结构体
type lru struct {
  //缓存最大数量
  k int
  // 存放节点数据
  cache map[string]*node
  //头节点
  head *node
  // 尾节点
  tail *node
}
// 工厂方法,构造一个LRU缓存对象
func NewLRUCache(k int) *lru {
  return &lru{k: k}
}
// 对外开放的Get方法
func (lru *lru) Get(key string) string {
  if lru.cache == nil {
    return ""
  }
  n, ok := lru.cache[key]
  if ok {
    value := n.value
    //更新到头节点
    lru.moveNodeToHead(n)
    return value
  } else {
    return ""
  }
}
// 对外开放的Set方法
func (lru *lru) Set(key string, value string) {
  //初始化map
  if lru.cache == nil {
    lru.cache = make(map[string]*node)
  }
  n, ok := lru.cache[key]
  if ok {
    //更新缓存
    n.value = value
  } else {
    //创建新节点
    n = &node{key: key, value: value}
    // 添加到map中
    lru.cache[key] = n
    //如果超过缓存大小,移除最后一个节点
    if len(lru.cache) > lru.k {
      lru.removeTail()
    }
  }
  // 更新节点为头节点
  lru.moveNodeToHead(n)
}
//将指定节点移到头节点
func (lru *lru) moveNodeToHead(n *node) {
  //如果头节点和尾节点都是空,说明是第一次添加数据
  if lru.head == nil && lru.tail == nil {
    lru.head = n
    lru.tail = n
    return
  }
  // 如果是头节点,那么不动
  if lru.head == n {
    return
  }
  nextNode := n.next
  prevNode := n.prev
  // 如果是尾节点,那么要先与上一个节点断开
  if n == lru.tail {
    prevNode.next = nil
    n.prev = nil
  } else if prevNode != nil && nextNode != nil {
    //如果是中间节点,要与前后节点都断开
    nextNode.prev = prevNode
    prevNode.next = nextNode
    n.prev = nil
  }
  //把节点移到到头节点
  n.next = lru.head
  lru.head.prev = n
  lru.head = n
}
//移除最后一个节点
func (lru *lru) removeTail() {
  // 如果尾节点wei空,那么清空数据直接返回
  if lru.tail == nil {
    lru.cache = nil
    return
  }
  tailPrev := lru.tail.prev
  // 尾节点与上一个节点断开
  lru.tail.prev = nil
  if tailPrev != nil {
    tailPrev.next = nil
  }
  // 把上一个节点置为尾节点
  lru.tail = tailPrev
  // 从map中删除尾节点数据
  delete(lru.cache, lru.tail.key)
}
// 重写String方法,自定义打印结果
func (lru *lru) String() string {
  point := lru.head
  result := ""
  flag := "\t"
  for point != nil {
    if len(result) > 0 {
      result = fmt.Sprintf("%v%v[%v:%v]", result, flag, point.key, point.value)
    } else {
      result = fmt.Sprintf("[%v:%v]", point.key, point.value)
    }
    point = point.next
  }
  return result
}
func main() {
  LRUCache := NewLRUCache(5)
  LRUCache.Set("1", "1")
  LRUCache.Set("2", "2")
  LRUCache.Set("3", "3")
  LRUCache.Set("4", "4")
  LRUCache.Set("5", "5")
  //打印缓存数据,验证容量是否起作用
  fmt.Println(LRUCache)
  //设置超过最大容量的数据
  LRUCache.Set("6", "6")
  //打印缓存数据,验证超过容量是否起作用
  fmt.Println(LRUCache)
  //查询头节点数据
  fmt.Println(LRUCache.Get("6"))
  //打印缓存数据
  fmt.Println(LRUCache)
  //查询非头节点数据
  fmt.Println(LRUCache.Get("4"))
  //打印缓存数据
  fmt.Println(LRUCache)
  //查询尾节点数据
  fmt.Println(LRUCache.Get("2"))
  //打印缓存数据
  fmt.Println(LRUCache)
}
复制代码

e15a94d147874b11b8e67c8a6ec844fc_tplv-k3u1fbpfcp-zoom-in-crop-mark_4536_0_0_0.png


相关文章
|
5天前
|
存储 Go
Go 语言入门指南:切片
Golang中的切片(Slice)是基于数组的动态序列,支持变长操作。它由指针、长度和容量三部分组成,底层引用一个连续的数组片段。切片提供灵活的增减元素功能,语法形式为`[]T`,其中T为元素类型。相比固定长度的数组,切片更常用,允许动态调整大小,并且多个切片可以共享同一底层数组。通过内置的`make`函数可创建指定长度和容量的切片。需要注意的是,切片不能直接比较,只能与`nil`比较,且空切片的长度为0。
Go 语言入门指南:切片
|
8天前
|
算法 安全 Go
公司局域网管理系统里的 Go 语言 Bloom Filter 算法,太值得深挖了
本文探讨了如何利用 Go 语言中的 Bloom Filter 算法提升公司局域网管理系统的性能。Bloom Filter 是一种高效的空间节省型数据结构,适用于快速判断元素是否存在于集合中。文中通过具体代码示例展示了如何在 Go 中实现 Bloom Filter,并应用于局域网的 IP 访问控制,显著提高系统响应速度和安全性。随着网络规模扩大和技术进步,持续优化算法和结合其他安全技术将是企业维持网络竞争力的关键。
24 2
公司局域网管理系统里的 Go 语言 Bloom Filter 算法,太值得深挖了
|
4天前
|
开发框架 前端开发 Go
eino — 基于go语言的大模型应用开发框架(二)
本文介绍了如何使用Eino框架实现一个基本的LLM(大语言模型)应用。Eino中的`ChatModel`接口提供了与不同大模型服务(如OpenAI、Ollama等)交互的统一方式,支持生成完整响应、流式响应和绑定工具等功能。`Generate`方法用于生成完整的模型响应,`Stream`方法以流式方式返回结果,`BindTools`方法为模型绑定工具。此外,还介绍了通过`Option`模式配置模型参数及模板功能,支持基于前端和用户自定义的角色及Prompt。目前主要聚焦于`ChatModel`的`Generate`方法,后续将继续深入学习。
82 6
|
5天前
|
存储 开发框架 Devops
eino — 基于go语言的大模型应用开发框架(一)
Eino 是一个受开源社区优秀LLM应用开发框架(如LangChain和LlamaIndex)启发的Go语言框架,强调简洁性、可扩展性和可靠性。它提供了易于复用的组件、强大的编排框架、简洁明了的API、最佳实践集合及实用的DevOps工具,支持快速构建和部署LLM应用。Eino不仅兼容多种模型库(如OpenAI、Ollama、Ark),还提供详细的官方文档和活跃的社区支持,便于开发者上手使用。
58 6
|
5天前
|
存储 算法 Go
Go语言实战:错误处理和panic_recover之自定义错误类型
本文深入探讨了Go语言中的错误处理和panic/recover机制,涵盖错误处理的基本概念、自定义错误类型的定义、panic和recover的工作原理及应用场景。通过具体代码示例介绍了如何定义自定义错误类型、检查和处理错误值,并使用panic和recover处理运行时错误。文章还讨论了错误处理在实际开发中的应用,如网络编程、文件操作和并发编程,并推荐了一些学习资源。最后展望了未来Go语言在错误处理方面的优化方向。
|
5天前
|
网络协议 算法 安全
Go语言的网络编程与TCP_UDP
Go语言由Google开发,旨在简单、高效和可扩展。本文深入探讨Go语言的网络编程,涵盖TCP/UDP的基本概念、核心算法(如滑动窗口、流量控制等)、最佳实践及应用场景。通过代码示例展示了TCP和UDP的实现,并讨论了其在HTTP、DNS等协议中的应用。最后,总结了Go语言网络编程的未来发展趋势与挑战,推荐了相关工具和资源。
|
7天前
|
缓存 监控 算法
内网监控管理软件:PHP 语言队列算法揭秘
在数字化办公环境中,内网监控管理软件对企业的稳定运行和信息安全至关重要。本文深入介绍PHP中的队列算法及其在内网监控软件中的应用,包括监控数据收集、任务调度和日志记录等场景,通过代码示例展示其实现方法。队列算法可提高性能、保证数据顺序并实现异步处理,为企业提供高效的安全保障。
18 1
|
13天前
|
存储 监控 算法
探秘员工泄密行为防线:基于Go语言的布隆过滤器算法解析
在信息爆炸时代,员工泄密行为对企业构成重大威胁。本文聚焦布隆过滤器(Bloom Filter)这一高效数据结构,结合Go语言实现算法,帮助企业识别和预防泄密风险。通过构建正常操作“指纹库”,实时监测员工操作,快速筛查可疑行为。示例代码展示了如何利用布隆过滤器检测异常操作,并提出优化建议,如调整参数、结合日志分析系统等,全方位筑牢企业信息安全防线,守护核心竞争力。
|
1天前
|
SQL 安全 Java
阿里双十一背后的Go语言实践:百万QPS网关的设计与实现
解析阿里核心网关如何利用Go协程池、RingBuffer、零拷贝技术支撑亿级流量。 重点分享: ① 如何用gRPC拦截器实现熔断限流; ② Sync.Map在高并发读写中的取舍。
|
2天前
|
存储 算法 安全
基于 Go 语言的公司内网管理软件哈希表算法深度解析与研究
在数字化办公中,公司内网管理软件通过哈希表算法保障信息安全与高效管理。哈希表基于键值对存储和查找,如用户登录验证、设备信息管理和文件权限控制等场景,Go语言实现的哈希表能快速验证用户信息,提升管理效率,确保网络稳定运行。
11 0