利用OPT算法解决最短访问次数问题

一、题目

数据库缓存,模拟访问规则如下:
当查询值在缓存中,直接访问缓存,不访问数据库。否则,访问数据库,并将值放入缓存。
若缓存已满,则必须删除一个缓存。
给定缓存大小和训练数据,依次模拟访问这组数据,请分析在此访问规则下最少的数据库访问次数
输入
2
1 2 3 1 2 3
输出
4

二、代码实现

/**
     * 数据库缓存
     * 模拟访问规则如下:
     * 当查询值在缓存中,直接访问缓存,不访问数据库。否则,访问数据库,并将值放入缓存。
     * 若缓存已满,则必须删除一个缓存。
     * 给定缓存大小和训练数据,依次模拟访问这组数据,请分析在此访问规则下最少的数据库访问次数
     * 输入
     * 2
     * 1 2 3 1 2 3
     * 输出
     * 4
     */
    public static int queryDataBase(int cacheSize,int[] ids){
        // 操作系统OPT算法,最优的删除策略
        // 如果当前缓存中某个元素在之后不会被使用到,那么可以将其删除
        // 如果当前缓存中所有元素在之后的都会使用到,那么删除“最晚使用到”的那个元素
        // 这样才能使得访问数据库总数最少
        if(ids.length==0){
            return 0;
        }
        HashSet<Integer> cache = new HashSet<>();
        int queryCount = 0;
        for (int i = 0; i < ids.length; i++) {
            if(cache.contains(ids[i])){
                continue;
            }
            
            if(cache.size() < cacheSize){
                // 缓存空间有剩余
                cache.add(ids[i]);
                queryCount++;
            }else{
                // 缓存空间已满
                queryCount++;
                int delNum = getDelNum(cache,ids,i+1);
                cache.remove(delNum);
                cache.add(ids[i]);
            }
        }
        return queryCount;
    }

    // 在从start之后的ids中找到需要删除的元素
    private static int getDelNum(HashSet<Integer> cache, int[] ids, int start) {
        List<Integer> remainList = new ArrayList<>();
        for (int i = start; i < ids.length; i++) {
            remainList.add(ids[i]);
        }

        // 优先返回缓存的值在后续数组中不会再出现的值
        for (Integer num : cache) {
            if(!remainList.contains(num)){
                return num;
            }
        }

        // 删除“最晚使用到”的那个元素
        int lastIndex = 0;
        for (Integer num : cache) {
            for (int i = 0; i < remainList.size(); i++) {
                if(remainList.get(i) == num){
                    lastIndex = Math.max(lastIndex,i);
                    break;
                }
            }
        }
        return remainList.get(lastIndex);
    }
  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值