加入收藏 | 设为首页 | 会员中心 | 我要投稿 南平站长网 (https://www.0599zz.com/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 运营中心 > 交互 > 正文

redis lru实现策略

发布时间:2016-11-01 04:34:41 所属栏目:交互 来源:站长网
导读:副标题#e# 在使用redis作为缓存的场景下,内存淘汰策略决定的redis的内存使用效率。在大部分场景下,我们会采用LRU(Least Recently Used)来作为redis的淘汰策略。本文将由浅入深的介绍redislru策略的具体实现。 首先我们来科普下,什么是LRU ?(以下来自维
副标题[/!--empirenews.page--]      在使用redis作为缓存的场景下,内存淘汰策略决定的redis的内存使用效率。在大部分场景下,我们会采用LRU(Least Recently Used)来作为redis的淘汰策略。本文将由浅入深的介绍redis lru策略的具体实现。
       首先我们来科普下,什么是LRU ?(以下来自维基百科) 
       Discards the least recently used items first. This algorithm requires keeping track of what was used when, which is expensive if one wants to make sure the algorithm always discards the least recently used item. General implementations of this technique require keeping "age bits" for cache-lines and track the "Least Recently Used"  cache-line based on age-bits. In such an implementation, every time a cache-line is used, the age of all other cache-lines changes.
      简而言之,就是每次淘汰最近最少使用的元素 。一般的实现,都是采用对存储在内存的元素采用 'age bits’ 来标记该元素从上次访问到现在为止的时长,从而在每次用LRU淘汰时,淘汰这些最长时间未被访问的元素。
    
      这里我们先实现一个简单的LRU Cache,以便于后续内容的理解 。(来自leetcod,不过这里我重新用Python语言实现了)       实现该缓存满足如下两点:
      1.get(key) - 如果该元素(总是正数)存在,将该元素移动到lru头部,并返回该元素的值,否则返回-1。
      2.set(key,value) - 设置一个key的值为value(如果该元素存在),并将该元素移动到LRU头部。否则插入一个key,且值为value。如果在设置前检查到,该key插入后,会超过cache的容量,则根据LRU策略,删除最近最少使用的key。
       分析
      这里我们采用双向链表来实现元素(k-v键值对)的存储,同时采用hash表来存储相关的key与item的对应关系。这样,我们既能在O(1)的时间对key进行操作,同时又能利用Double LinkedList的添加和删除节点的便利性。(get/set都能在O(1)内完成)。

    redis lru实现策略
 具体实现(Python语言)

点击(此处)折叠或打开

  1. class Node:
  2.       key=None
  3.       value=None
  4.       pre=None
  5.       next=None

  6. def __init__(self,key,value):
  7.       self.key=key
  8.       self.value=value

  9. class LRUCache:
  10.       capacity=0
  11.       map={} # key is string ,and value is Node object
  12.       head=None
  13.       end=None

  14. def __init__(self,capacity):
  15.       self.capacity=capacity

  16. def get(self,key):
  17.       if key in self.map:
  18.            node=self.map[key]
  19.            self.remove(node)
  20.            self.setHead(node)
  21.            return node.value
  22.       else:
  23.           return -1

  24. def getAllKeys(self):
  25.        tmpNode=None
  26.        if self.head:
  27.           tmpNode=self.head
  28.           while tmpNode:
  29.           print (tmpNode.key,tmpNode.value)
  30.           tmpNode=tmpNode.next

  31. def remove(self,n):
  32.       if n.pre:
  33.          n.pre.next=n.next
  34.       else:
  35.          self.head=n.next

  36.       if n.next:
  37.          n.next.pre=n.pre
  38.       else:
  39.          self.end=n.pre

  40. def setHead(self,n):
  41.       n.next=self.head
  42.       n.pre=None

  43.       if self.head:
  44.          self.head.pre=n

  45.       self.head=n

  46.       if not self.end:
  47.          self.end=self.head

  48. def set(self,key,value):
  49.        if key in self.map:
  50.           oldNode=self.map[key]
  51.           oldNode.value=value
  52.           self.remove(oldNode)
  53.           self.setHead(oldNode)
  54.        else:
  55.           node=Node(key,value)
  56.           if len(self.map) >= self.capacity:
  57.              self.map.pop(self.end.key)
  58.              self.remove(self.end)
  59.              self.setHead(node)
  60.           else:
  61.              self.setHead(node)

  62.           self.map[key]=node


  63. def main():
  64.        cache=LRUCache(100)

  65.        #d->c->b->a
  66.        cache.set('a','1')
  67.        cache.set('b','2')
  68.        cache.set('c',3)
  69.        cache.set('d',4)

  70.        #遍历lru链表
  71.        cache.getAllKeys()

  72.        #修改('a','1') ==> ('a',5),使该节点从LRU尾端移动到开头.
  73.        cache.set('a',5)
  74.        #LRU链表变为 a->d->c->b

  75.        cache.getAllKeys()
  76.        #访问key='c'的节点,是该节点从移动到LRU头部
  77.        cache.get('c')
  78.        #LRU链表变为 c->a->d->b
  79.        cache.getAllKeys()

  80.        if __name__ == '__main__':
  81.           main()
     通过上面简单的介绍与实现,现在我们基本已经了解了什么是LRU,下面我们来看看LRU算法在redis 内部的实现细节,以及其会在什么情况下带来问题。在redis内部,是通过全局结构体struct redisServer 保存redis启动之后相关的信息,比如:

(编辑:南平站长网)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

热点阅读