【毎日の練習] - 使用頻度の最も低い(LFU)キャッシュ・データ構造

タイトル説明

設計し、最も頻繁に使用される(LFU)キャッシュデータ構造を実装しています。これは、次の操作をサポートする必要があります。getおよびput。
(キー)を取得-キーがキャッシュに存在する場合は、その値(常に正)キーを取得し、それ以外の場合は-1。
プット(キー、値) -キーが存在し、またはセット値を挿入していない場合。キャッシュがその容量に達した場合、それは、新しいプロジェクトを挿入する前にする必要があり、このプロジェクトは、最も頻繁に使用されるが無効ではありません。この問題では、ときにタイがある(すなわち、同一の周波数を有する2つ以上のキー)、最も最近使用されたキーが除去されます。
タイトルはleetcodeから来て、入力します

読むのタイトル

例:

LFUCache cache = new LFUCache( 2 /* capacity (缓存容量) */ );

cache.put(1, 1);
cache.put(2, 2);
cache.get(1);       // 返回 1
cache.put(3, 3);    // 去除 key 2
cache.get(2);       // 返回 -1 (未找到key 2)
cache.get(3);       // 返回 3
cache.put(4, 4);    // 去除 key 1
cache.get(1);       // 返回 -1 (未找到 key 1)
cache.get(3);       // 返回 3
cache.get(4);       // 返回 4

次のように設計されています:
ここに画像を挿入説明

コード

LFUCacheクラス

public class LFUCache {
    private int cap; // 缓存总容量
    private int size; // 当前缓存使用量
    private HashMap<Integer, Node> cache; // 缓存Map
    private HashMap<Integer, ArrayList<Node>> freqNodes; // 记录使用次数

    // 打印当前缓存内容 测试用
    public void printCache(){
        for (Map.Entry<Integer, Node> entry : cache.entrySet()){
            System.out.print(entry.getKey()+" ");
        }
        System.out.println();
    }
    public LFUCache(int capacity) {
        this.cap = capacity;
        this.cache = new HashMap<Integer, Node>(capacity);
        this.freqNodes = new HashMap<Integer, ArrayList<Node>>();
    }

    public int get(int key) {
        Node node = cache.get(key);
        if (node == null) {
            return -1;
        }
        node.freq++;
        // 次数有变化,需要更新记录使用次数的Map
        adjustPosition(node);
        return node.value;
    }

    private void adjustPosition(Node node) {
        if (freqNodes.containsKey(node.freq - 1)) {
            freqNodes.get(node.freq - 1).remove(node);
            // list为空时需要将它从记录使用次数的map中移除
            if (freqNodes.get(node.freq - 1).size() == 0) {
                freqNodes.remove(node.freq - 1);
            }
        }
        // 将node加到新的list,如果不存在需要新建
        if (freqNodes.containsKey(node.freq)) {
            freqNodes.get(node.freq).add(node);
        } else {
            ArrayList<Node> newNodes = new ArrayList<Node>();
            newNodes.add(node);
            freqNodes.put(node.freq, newNodes);
        }

    }


    public int getMinReq() {
        int req  = 0;
        boolean flag = true;
        // 循环获取最小key,没有什么好说的
        for (Map.Entry<Integer, ArrayList<Node>> entry : freqNodes.entrySet()) {
            if(flag){
                req = entry.getKey();
                flag = false;
            }else{
                req = Integer.min(req, entry.getKey());
            }
        }
        return rep;
    }

    public void put(int key, int value) {
        // 容量为0无法存储
        if (cap == 0) return;
        // 判断是否已经存在
        Node node = cache.get(key);

        if (node == null) {
            // 不存在的情况,需要判断缓存是否已满,没有满,直接加入,若满了,需要找到使用次数最少最近调用时间最靠后的
            if (size == cap) {
            	// 找到使用次数最少且最近一次使用时间最长的元素
                int remove = getMinReq();
                int k = freqNodes.get(remove).get(0).key;
                // 将找到的元素从原来的list中删除
                freqNodes.get(remove).remove(0);
                // 从缓存中删除
                cache.remove(k);
                node = new Node(key,value);
                // 新增时,需要更新记录使用次数的Map
                adjustPosition(node);
                cache.put(key, node);
            } else {
                Node newNode = new Node(key, value);
                // 新增时,需要更新记录使用次数的Map
                adjustPosition(newNode);// 加入缓存
                cache.put(key, newNode);
                // 当前缓存大小增加
                size++;
            }

        } else {
            // 已存在的情况,更新调用次数,更新值
            node.value = value;
            node.freq++;
            // 次数有变化,需要更新记录使用次数的Map
            adjustPosition(node);
        }
    }


}

Node 类

class Node {
    int key;
    int value;
    int freq = 0;

    public Node(int key, int value) {
        this.key = key;
        this.value = value;
    }

    public Node() {

    }
}

テスト

public class Test {
    public static void main(String[] args) {
        LFUCache lfuCache = new LFUCache(5);
        lfuCache.put(1,1);
        lfuCache.put(2,2);
        lfuCache.put(3,3);
        lfuCache.put(4,4);
        lfuCache.put(5,5);
        System.out.println(lfuCache.get(1));
        System.out.println(lfuCache.get(2));
        System.out.println(lfuCache.get(4));
        System.out.println(lfuCache.get(5));
        lfuCache.put(6,6);
        System.out.println("push(6,6)此时cache:");
        lfuCache.printCache();
        System.out.println(lfuCache.get(3));
        System.out.println(lfuCache.get(6));
        System.out.println("================此时剩余12456 均只调用一次 1距离上次调用时间最久================");
        lfuCache.put(7,7); // 此时应该为 24567
        System.out.println("push(7,7),此时cache:");
        lfuCache.printCache();
        System.out.println(lfuCache.get(1));// 应为 -1


    }
}
  • 結果:
    ここに画像を挿入説明
公開された32元の記事 ウォン称賛7 ビュー4075

おすすめ

転載: blog.csdn.net/weixin_45676630/article/details/105326871