博客
关于我
LRU的map+双链表实现(Go描述)
阅读量:749 次
发布时间:2019-03-22

本文共 2071 字,大约阅读时间需要 6 分钟。

面云账户时候问了LRU,具体实现的方式是map+双链表。Set和Get的时间复杂度都是O(1)。完整写一遍复习一下, 仅作记录

/** * @Author: lzw5399 * @Date: 2021/5/20 22:28 * @Desc: 基于map和双链表实现的LRU算法 */package mainimport "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)}// LRUCachetype 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}

转载地址:http://qnkwk.baihongyu.com/

你可能感兴趣的文章
mysql generic安装_MySQL 5.6 Generic Binary安装与配置_MySQL
查看>>
Mysql group by
查看>>
MySQL I 有福啦,窗口函数大大提高了取数的效率!
查看>>
mysql id自动增长 初始值 Mysql重置auto_increment初始值
查看>>
MySQL in 太多过慢的 3 种解决方案
查看>>
MySQL InnoDB 三大文件日志,看完秒懂
查看>>
Mysql InnoDB 数据更新导致锁表
查看>>
Mysql Innodb 锁机制
查看>>
MySQL InnoDB中意向锁的作用及原理探
查看>>
MySQL InnoDB事务隔离级别与锁机制深入解析
查看>>
Mysql InnoDB存储引擎 —— 数据页
查看>>
Mysql InnoDB存储引擎中的checkpoint技术
查看>>
Mysql InnoDB存储引擎中缓冲池Buffer Pool、Redo Log、Bin Log、Undo Log、Channge Buffer
查看>>
MySQL InnoDB引擎的锁机制详解
查看>>
Mysql INNODB引擎行锁的3种算法 Record Lock Next-Key Lock Grap Lock
查看>>
mysql InnoDB数据存储引擎 的B+树索引原理
查看>>
mysql innodb通过使用mvcc来实现可重复读
查看>>
mysql insert update 同时执行_MySQL进阶三板斧(三)看清“触发器 (Trigger)”的真实面目...
查看>>
mysql interval显示条件值_MySQL INTERVAL关键字可以使用哪些不同的单位值?
查看>>
Mysql join原理
查看>>