一种具有O(1)复杂度的LFU算法实现(java 实现)

文章介绍了如何使用双向链表和HashMap实现一个具有O(1)复杂度的LFU(LeastFrequentlyUsed)数据缓存算法。在该实现中,数据节点存储在一个基于访问频率的双向链表中,当缓存满时,会优先淘汰访问频率最低的元素。文章详细阐述了添加、获取和删除操作的过程。
摘要由CSDN通过智能技术生成

最近在做一些数据缓存方面的工作。研究了LRU算法和LFU算法。

算法本质上是维护一个如图的数据结构:

  • 一个HashMap
  • 一个双向链表(链表元素又是一个双向子链表,具有相同访问次数的数据会放在相同的链表中)。
    在这里插入图片描述

此时聪明的你应该想到如何实现了!下面以一个例子说明其原理。为了画图方便。省略了两类指针:

  1. HashMap指向子链表元素的指针
  2. 双向子链表元素指向双向链表节点的指针

在本文的实现中,初始时只有一个频次为0的链表:
在这里插入图片描述

当加入值时,都默认放在这个频次为0的链表中。比如添加a, b, c后。
在这里插入图片描述

当获取值时,会把它的频次加1,以b为例,即会把b移动到一个新的链表。
在这里插入图片描述

当缓存不足时,将优先淘汰访问频次最小的。可以从head开始查找,将找到的第一个从链表中删除即可。即会把a删除,删除后如图。
在这里插入图片描述

无论是加入、获取、淘汰,操作的复杂度都是O(1)的。

package com.iscas.intervention.data.cache;

import java.util.HashMap;

/**
 * O(1) 的LFU实现
 * @author cloudea
 * @date 2023/03/17
 */
public class LFU {

    /**
     * 保存数据的双向链表结点
     */
    public  static class LFUListNode {
        public String key;
        public Object value;
        public LFUListNode last;
        public LFUListNode next;

        public LFUDoubleListNode master; // 指向所属的频次节点

        public LFUListNode(String key, Object value) {
            this.key = key;
            this.value = value;
        }
    }

    /**
     * 保存频次的双向链表节点
     */
    public static class LFUDoubleListNode {
        public int count; // 频次
        public LFUListNode head ;
        public LFUListNode tail;
        public LFUDoubleListNode last;
        public LFUDoubleListNode next;

        public LFUDoubleListNode (int count){
            this.count = count;
            // 初始化哨兵
            this.head = new LFUListNode(null, null);
            this.tail = new LFUListNode(null, null);
            this.head.next = this.tail;
            this.tail.last = this.head;
        }
    }

    /**
     * 频次双向链表
     */
    public static class LFUDoubleList {
        public LFUDoubleListNode head;
        public LFUDoubleListNode tail;

        public LFUDoubleList(){
            this.head = new LFUDoubleListNode(0);
            this.tail = new LFUDoubleListNode(0);
            this.head.next = this.tail;
            this.tail.last = this.head;
        }

    }


    private HashMap<String, LFUListNode> caches = new HashMap<>();
    private LFUDoubleList counts = new LFUDoubleList();


    public LFU(){
        // 初始化访问频次为0的结点
        LFUDoubleListNode lfuDoubleListNode = new LFUDoubleListNode(0);
        lfuDoubleListNode.last = counts.head;
        lfuDoubleListNode.next = counts.tail;
        counts.head.next = lfuDoubleListNode;
        counts.tail.last = lfuDoubleListNode;
    }

    public void set(String key, Object value){
        if(caches.containsKey(key)) {
            // 如果存在,则更新一下value即可
            caches.get(key).value = value;
        }else{
            // 否则,需要放入链表中(放入频率为0的链表)
            LFUListNode node = new LFUListNode(key, value);
            node.master = counts.head.next;
            node.last = node.master.tail.last;
            node.next = node.master.tail;
            node.last.next = node;
            node.next.last = node;
            caches.put(key, node);
        }
    }

    public Object get(String key){
        if(caches.containsKey(key)){
            // 如果存在,直接返回,然后更新频次 (频次加1)
            LFUListNode node = caches.get(key);
            LFUDoubleListNode nextMaster = node.master.next;
            if(nextMaster.count != node.master.count + 1) {
                // 新建一个master节点
                LFUDoubleListNode A = node.master;
                LFUDoubleListNode B = node.master.next;
                LFUDoubleListNode n = new LFUDoubleListNode(node.master.count + 1);
                n.last = A;
                n.next = B;
                A.next = n;
                B.last = n;
                nextMaster = n;
            }
            // 把结点从当前master删除
            node.last.next = node.next;
            node.next.last = node.last;

            // 把结点加入到下一个master
            node.last = nextMaster.tail.last;
            node.next = nextMaster.tail;
            node.last.next = node;
            node.next.last = node;
            node.master = nextMaster;

            // 返回节点值
            return node.value;
        }
        return null;
    }

    /**
     * 删除频次最小的元素
     */
    public Object remove(){
        if(size() != 0){
            for(var i = counts.head.next; i != counts.tail; i = i.next){
                for(var j = i.head.next; j != i.tail; j = j.next){
                    // 删除当前元素
                    j.last.next = j.next;
                    j.next.last = j.last;
                    // 删除当前master(如果没有结点的话)
                    if(j.master.head.next == j.master.tail){
                        if(j.master.count != 0){
                            j.master.last.next = j.master.next;
                            j.master.next.last = j.master.last;
                        }
                    }
                    // 从hashmap中删除
                    caches.remove(j.key);
                    return j.value;
                }
            }
        }
        return null;
    }

    public int size(){
        return caches.size();
    }
}

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值