JS 实现缓存算法(FIFO/LRU)

FIFO

最简单的一种缓存算法,设置缓存上限,当达到了缓存上限的时候,按照先进先出的策略进行淘汰,再增长进新的 k-v 。node

使用了一个对象做为缓存,一个数组配合着记录添加进对象时的顺序,判断是否到达上限,若到达上限取数组中的第一个元素key,对应删除对象中的键值。算法

/**
 * FIFO队列算法实现缓存
 * 须要一个对象和一个数组做为辅助
 * 数组记录进入顺序
 */
class FifoCache{
    constructor(limit){
        this.limit = limit || 10
        this.map = {}
        this.keys = []
    }
    set(key,value){
        let map = this.map
        let keys = this.keys
        if (!Object.prototype.hasOwnProperty.call(map,key)) {
            if (keys.length === this.limit) {
                delete map[keys.shift()]//先进先出,删除队列第一个元素
            }
            keys.push(key)
        }
        map[key] = value//不管存在与否都对map中的key赋值
    }
    get(key){
        return this.map[key]
    }
}

module.exports = FifoCache

LRU

LRU(Least recently used,最近最少使用)算法。该算法的观点是,最近被访问的数据那么它未来访问的几率就大,缓存满的时候,优先淘汰最无人问津者数组

算法实现思路:基于一个双链表的数据结构,在没有满员的状况下,新来的 k-v 放在链表的头部,之后每次获取缓存中的 k-v 时就将该k-v移到最前面,缓存满的时候优先淘汰末尾的。缓存

双向链表的特色,具备头尾指针,每一个节点都有 prev(前驱) 和 next(后继) 指针分别指向他的前一个和后一个节点。数据结构

关键点:在双链表的插入过程当中要注意顺序问题,必定是在保持链表不断的状况下先处理指针,最后才将原头指针指向新插入的元素,在代码的实现中请注意看我在注释中说明的顺序注意点!this

class LruCache {
    constructor(limit) {
        this.limit = limit || 10
        //head 指针指向表头元素,即为最经常使用的元素
        this.head = this.tail = undefined
        this.map = {}
        this.size = 0
    }
    get(key, IfreturnNode) {
        let node = this.map[key]
        // 若是查找不到含有`key`这个属性的缓存对象
        if (node === undefined) return
        // 若是查找到的缓存对象已是 tail (最近使用过的)
        if (node === this.head) { //判断该节点是否是是第一个节点
            // 是的话,皆大欢喜,不用移动元素,直接返回
            return returnnode ?
                node :
                node.value
        }
        // 不是头结点,铁定要移动元素了
        if (node.prev) { //首先要判断该节点是否是有前驱
            if (node === this.tail) { //有前驱,如果尾节点的话多一步,让尾指针指向当前节点的前驱
                this.tail = node.prev
            }
            //把当前节点的后继交接给当前节点的前驱去指向。
            node.prev.next = node.next
        }
        if (node.next) { //判断该节点是否是有后继
            //有后继的话直接让后继的前驱指向当前节点的前驱
            node.next.prev = node.prev
            //整个一个过程就是把当前节点拿出来,而且保证链表不断,下面开始移动当前节点了
        }
        node.prev = undefined //移动到最前面,因此没了前驱
        node.next = this.head //注意!!! 这里要先把以前的排头给接到手!!!!让当前节点的后继指向原排头
        if (this.head) {
            this.head.prev = node //让以前的排头的前驱指向如今的节点
        }
        this.head = node //完成了交接,才能执行此步!否则就找不到以前的排头啦!
        return IfreturnNode ?
            node :
            node.value
    }
    set(key, value) {
        // 以前的算法能够直接存k-v可是如今要把简单的 k-v 封装成一个知足双链表的节点
        //1.查看是否已经有了该节点
        let node = this.get(key, true)
        if (!node) {
            if (this.size === this.limit) { //判断缓存是否达到上限
                //达到了,要删最后一个节点了。
                if (this.tail) {
                    this.tail = this.tail.prev
                    this.tail.prev.next = undefined
                    //平滑断链以后,销毁当前节点
                    this.tail.prev = this.tail.next = undefined
                    this.map[this.tail.key] = undefined
                    //当前缓存内存释放一个槽位
                    this.size--
                }
                node = {
                    key: key
                }
                this.map[key] = node
                if(this.head){//判断缓存里面是否是有节点
                    this.head.prev = node
                    node.next = this.head
                }else{
                    //缓存里没有值,皆大欢喜,直接让head指向新节点就好了
                    this.head = node
                    this.tail = node
                }
                this.size++//减小一个缓存槽位
            }
        }
        //节点存不存在都要给他从新赋值啊
        node.value = value
    }
}

module.exports = LruCache

具体的思路就是若是所要get的节点不是头结点(即已是最近使用的节点了,不须要移动节点位置)要先进行平滑的断链操做,处理好指针指向的关系,拿出须要移动到最前面的节点,进行链表的插入操做。prototype

相关文章
相关标签/搜索