缓存中间件(一)

缓存中间件的概念并不难理解,我们知道数据被存储在内存中时查询速度会非常慢,而查询缓存中的数据的查询时间几乎可以忽略不计。为何不将数据全部存储在缓存中呢?因为缓存的容量小。

缓存中间件就像书桌旁的抽屉

  • 把每天都要用的东西(高频读取的数据)放抽屉里,不用每次都跑大仓库(数据库)找,拿取更快;
  • 只有抽屉里没有(缓存未命中),才去仓库拿,顺便把东西放回抽屉(写入缓存),下次用更方便

缓存中间件的设置,本质上是利用空间换时间的平衡。现在我们利用java代码来模拟缓存中间件的运行逻辑。

java代码模拟缓存中间件:

1.创建实体类对象:
java 复制代码
public class User {
    private String name;
    private String phone;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getPhone() {
        return phone;
    }

    public void setPhone(String phone) {
        this.phone = phone;
    }
}

我们创建了一个User类,设置了两个属性:name和phone及其get、set方法。

2.创建CacheManger类来模拟缓存中间键:
java 复制代码
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
//模拟缓存中间件
public class CacheManger {
    //所有的缓存都是key-value形式键值对的形式
    //简单的key-value形式的缓存
    private static Map<String,String>cache = new HashMap<>();

    //处理简单形式的缓存
    //新增缓存
    public static void put(String key,String value){
        cache.put(key,value);
    }
    //查询缓存
    public static String get(String key){
        return cache.get(key);
    }
    //删除缓存
    public static void remove(String key){
        cache.remove(key);
    }

}

我们看到CacheManger对象中设置了一个Map类型的对象,这是缓存通常使用的基本类型(key-value键值对类型)。

处理方法也很简单:put、get和remove方法分别新增获取和删除键值对。

3.创建UserList类来模拟内存:
java 复制代码
import java.util.ArrayList;
import java.util.List;

public class UserList {
    //用来模拟从磁盘中找
    //模拟内存
    public static List<User>users = new ArrayList<User>();
    static {
        User user = new User();
        user.setName("张三");
        user.setPhone("123456");

        User user2 = new User();
        user2.setName("里斯");
        user2.setPhone("123");

        users.add(user);
        users.add(user2);
    }

    public static String getPhoneByUserName(String userName){
        for(User user : users){
            if(user.getName().equals(userName)){
                return user.getPhone();
            }
        }
        return null;
    }
}

这里static代码块内已经将"张三"和"里斯"两个键值对存储在模拟内存中了。

4.创建Main方法模拟缓存逻辑:
java 复制代码
import java.util.List;

public class Main {
    public static void main(String[] args) {
    //要查询张三的手机号
//        putAllCache();
        getPhoneByUserName("张三");
        getPhoneByUserName("里斯");
    }
    public static String getPhoneByUserName(String userName){
        //先查看缓存,缓存中没有再从列表中查
        String phone = CacheManger.get(userName);
        //如果缓存中没有,就从内存中找,从内存中找到后存放到缓存中间件中
        if(phone==null){
            System.out.println("缓存未命中");
            phone = UserList.getPhoneByUserName(userName);
            //放入缓存
            CacheManger.put(userName, phone);
        }
        System.out.println("缓存命中");
        return phone;
    }
    //缓存预热
    public static void putAllCache(){
        //将所有的UserList放到缓存中
        List<User> users = UserList.users;
        for(User user : users){
            CacheManger.put(user.getName(), user.getPhone());
        }
    }
}

缓存命中:表示缓存中没有想要查找的数据。

缓存未命中:表示缓存中没有想要查找的数据,代码会自动从内存中获取想要查找的值,并将其存贮在缓存中。

注意:不论缓存中是否有想要查找的值,缓存最后一定会命中。

由此,当执行此代码时,结果是:

因为一开始缓存中没有需要查找的相关数据,显示未命中。最后将从内存中查找到的数据存放在缓存中,结果就是想要查找的手机号以键值对的形式被存储在模拟缓存中,结果一定是缓存命中。

而当将 putAllCache();(缓存预热) 这一句注释取消后,结果就是:

可以推断出缓存预热就是将内存中所有存储的值放入缓存中。

相关推荐
kronos.荒2 小时前
LRUCache缓存实现
算法·缓存·哈希算法
AI精钢2 小时前
WSL 磁盘清理实战:从缓存清理到 ext4.vhdx 压缩回收空间
运维·windows·缓存·docker·wsl·devops·磁盘清理
修行者Java2 小时前
(六)从“缓存混乱难管控”到“Redis高效赋能”——Redis实战进阶指南
数据库·redis·缓存
23.2 小时前
缺页中断与页缓存:高效内存管理揭秘
缓存·面试
onebyte8bits2 小时前
NestJS 系列教程(十五):缓存体系设计 —— Redis、接口缓存与缓存三大问题解决方案
数据库·redis·后端·缓存·nestjs
筱顾大牛3 小时前
缓存更新策略
java·redis·缓存
難釋懷3 小时前
Redis主从-repl_backlog原理
数据库·redis·缓存
fengxin_rou3 小时前
redis主从和集群一致性、哨兵机制详解
java·开发语言·数据库·redis·缓存
不光头强3 小时前
integer修改缓存大小
缓存