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


相关文章
|
2天前
|
Go 索引
Go 语言切片(Slice)
Go 语言切片(Slice)
7 1
|
2天前
|
存储 Go Python
Go 语言结构体
Go 语言结构体
6 0
|
2天前
|
存储 Go
Go 语言指针
Go 语言指针
6 0
|
2天前
|
JSON Java Go
使用go语言中的内置库调试性能
【5月更文挑战第21天】本文介绍Go 语言提供了内置的 expvar 模块来输出度量数据,帮助定位性能瓶颈。与 pprof 不同,expvar 专注于应用的宏观状态,通过 HTTP 接口 `/debug/vars` 提供标准的 JSON 格式数据,包括自定义度量和内存统计等。通过 expvar,开发者可以轻松监控应用状态,如消息处理速率、内存使用等,而无需像 C++ 或 Java 那样手动实现。
18 0
使用go语言中的内置库调试性能
|
3天前
|
编译器 Go 索引
Go 语言数组
Go 语言数组
8 1
|
3天前
|
Go CDN
Go 语言变量作用域
Go 语言变量作用域
13 4
|
3天前
|
编译器 Go
Go 语言函数
Go 语言函数
13 7
|
3天前
|
自然语言处理 算法 关系型数据库
再谈go语言中字符转换效率问题
【5月更文挑战第20天】本文讨论了Go语言中类型转换的效率,特别是`byte`、`rune`和`string`之间的转换。`性能测试显示,从`[]byte`到`string`的转换,新版与旧版性能相当;但从`string`到`[]byte`,旧版快于新版两倍。此外,文章提到了Unicode校对算法(UCA)的版本差异可能带来的排序和大小写转换不一致问题,这在多语言环境中需要注意。
18 1
再谈go语言中字符转换效率问题
|
3天前
|
编译器 Go 索引
浅谈go语言中的符文字符处理工具
【5月更文挑战第20天】本文简述了Go 1.20之后的rune符文处理工具和函数,`unsafe`包新增了SliceData、String和StringData函数,支持直接将slice转换为array,明确了数组和结构体比较顺序。
18 1
浅谈go语言中的符文字符处理工具
|
4天前
|
Go
Go 语言循环语句
Go 语言循环语句
10 0