[redis]lru的实现和lru的模拟代码

前言

redis的lru是近似lru算法,其实就是每个key里面都有存,最后一次访问这个key的时间戳

24位

1.因为lru需要很多内存,额外的内存,所以用近似lru

2.那么我们就可以用随机采样法淘汰元素,而且处理方式是懒惰删除。

3.一旦发现内存超过阈值,maxmermory,那么就随机采样出五个key,淘汰最久的key,如果淘汰后还是超过,那么久继续淘汰直到内存低于maxmemory;

这个5可设置~

具体的lru实现代码

package com.dk.learndemo.algorithm;

import java.util.HashMap;
import java.util.Map;

/**
 * @author : pmdream
 * @description : LRUCache
 * @create : 2020/07/10
 */
public class LRUCache {

    /**
     时间复杂度:对于 put 和 get 都是 O(1)
     空间复杂度:O(capacity),因为哈希表和双向链表最多存储 capacity+1 个元素。*/
    /**
     * 双向链表的结构
     * */
    class DLinkedNode {
        int key;
        int value;
        DLinkedNode prev;
        DLinkedNode next;
        public DLinkedNode() {}
        public DLinkedNode(int _key, int _value) {key = _key; value = _value;}
    }

    //缓存Map有点像手动实现一个LinkedHashMap
    private Map<Integer, DLinkedNode> cache = new HashMap<Integer, DLinkedNode>();
    //size大小
    private int size;
    private int capacity;
    //注意定义头尾节点
    private DLinkedNode head, tail;

    /**
     * 初始化容量
     * */
    public LRUCache(int capacity) {
        this.size = 0;
        this.capacity = capacity;
        // 使用伪头部和伪尾部节点
        head = new DLinkedNode();
        tail = new DLinkedNode();
        //初始化头尾节点
        head.next = tail;
        tail.prev = head;
    }

    /**
     * get key得到node,
     * 把node挪动到头结点
     * 并返回node value
     *
     * 只不过更改了node的位置
     * key对应的还是没变
     * */
    public int get(int key) {
        DLinkedNode node = cache.get(key);
        if (node == null) {
            return -1;
        }
        // 如果 key 存在,先通过哈希表定位,再移到头部
        moveToHead(node);
        return node.value;
    }

    /**
     * put key和value
     * value可以看做一个值
     * 但是把它扔到一个node里面
     * 很巧妙的是,他的node里面放了KEY 和value 那么就直接能知道它对应的key了根据node
     * */
    public void put(int key, int value) {
        DLinkedNode node = cache.get(key);
        if (node == null) {
            // 如果 key 不存在,创建一个新的节点
            DLinkedNode newNode = new DLinkedNode(key, value);
            // 添加进哈希表
            cache.put(key, newNode);
            // 添加至双向链表的头部
            addToHead(newNode);
            ++size;
            if (size > capacity) {
                // 如果超出容量,删除双向链表的尾部节点
                DLinkedNode tail = removeTail();
                // 删除哈希表中对应的项
                cache.remove(tail.key);
                --size;
            }
        }
        else {
            // 如果 key 存在,先通过哈希表定位,再修改 value,并移到头部
            node.value = value;
            moveToHead(node);
        }
    }

    private void addToHead(DLinkedNode node) {
        //node节点指向head 双向指针操作
        node.prev = head;
        node.next = head.next;
        head.next.prev = node;
        head.next = node;
    }

    private void removeNode(DLinkedNode node) {
        node.prev.next = node.next;
        node.next.prev = node.prev;
    }

    //如果get存在,那么应该对节点挪动到头部
    private void moveToHead(DLinkedNode node) {
        removeNode(node);
        addToHead(node);
    }

    private DLinkedNode removeTail() {
        DLinkedNode res = tail.prev;
        removeNode(res);
        return res;
    }
}

总结

get操作

lru的写法代码上面,主要注意:

    class DLinkedNode {
        int key;
        int value;
        DLinkedNode prev;
        DLinkedNode next;
        public DLinkedNode() {}
        public DLinkedNode(int _key, int _value) {key = _key; value = _value;}
    }

注意:

//缓存Map有点像手动实现一个LinkedHashMap
private Map<Integer, DLinkedNode> cache = new HashMap<Integer, DLinkedNode>();
//size大小
private int size;
private int capacity;
//注意定义头尾节点
private DLinkedNode head, tail;

构造函数:// 初始化好 tail 和head


    /**
     * 初始化容量
     * */
    public LRUCache(int capacity) {
        this.size = 0;
        this.capacity = capacity;
        // 使用伪头部和伪尾部节点
        head = new DLinkedNode();
        tail = new DLinkedNode();
        //初始化头尾节点
        head.next = tail;
        tail.prev = head;
    }

获取元素:// 需要把key 弄到head中~

    /**
     * get key得到node,
     * 把node挪动到头结点
     * 并返回node value
     *
     * 只不过更改了node的位置
     * key对应的还是没变
     * */
    public int get(int key) {
        DLinkedNode node = cache.get(key);
        if (node == null) {
            return -1;
        }
        // 如果 key 存在,先通过哈希表定位,再移到头部
        moveToHead(node);
        return node.value;
    }

  //如果get存在,那么应该对节点挪动到头部
    private void moveToHead(DLinkedNode node) {
        removeNode(node);
        addToHead(node);
    }
    
    // 把这个节点移除
    private void removeNode(DLinkedNode node) {
        node.prev.next = node.next;
        node.next.prev = node.prev;
    }
    // 然后弄到head上面
    private void addToHead(DLinkedNode node) {
        //node节点指向head 双向指针操作
        // 先让node 指定好了位置
        node.prev = head;
        node.next = head.next;
        // 再更改head
        head.next.prev = node;
        head.next = node;
    }
    
    

put操作解析

    /**
     * put key和value
     * value可以看做一个值
     * 但是把它扔到一个node里面
     * 很巧妙的是,他的node里面放了KEY 和value 那么就直接能知道它对应的key了根据node
     * */
    public void put(int key, int value) {
        DLinkedNode node = cache.get(key);
        if (node == null) {
            // 如果 key 不存在,创建一个新的节点
            DLinkedNode newNode = new DLinkedNode(key, value);
            // 添加进哈希表
            cache.put(key, newNode);
            // 添加至双向链表的头部
            addToHead(newNode);
            ++size;
            if (size > capacity) {
                // 如果超出容量,删除双向链表的尾部节点
                DLinkedNode tail = removeTail();
                // 删除哈希表中对应的项
                cache.remove(tail.key);
                --size;
            }
        }
        else {
            // 如果 key 存在,先通过哈希表定位,再修改 value,并移到头部
            node.value = value;
            moveToHead(node);
        }
    }

主要看key不存在的情况。

如果超过容量需要把tail弄走,然后删除hash表中的tail

因为lru都是没有那么满,所以先加入,然后后淘汰也是没有问题滴

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值