从小白到架构师 | 缓存预热

缓存预热指的是在系统启动或上线前,提前将经常访问的数据加载到缓存中,以避免在系统刚启动时,用户大量请求直接访问数据库,导致数据库压力过大或系统性能下降。通过缓存预热,可以确保系统一上线就能提供快速的响应时间和良好的用户体验。

缓存预热的几种解决方案:

  • 手动预热:系统管理员在系统启动前,手动触发一些接口,将数据加载到缓存中。
  • 自动预热
  • 定时预热

缓存预热的几种解决方案

自动预热

系统启动时,自动加载预定义的一些数据到缓存中。

java 复制代码
public class CacheService {
    private final Map<String, Data> cache = new ConcurrentHashMap<>();
    private final Database database = new Database();

    public CacheService() {
        // 系统启动时自动预热
        autoPreheat();
    }

    private void autoPreheat() {
        // 自动预热常用数据
        cache.put("key1", database.getData("key1"));
        cache.put("key2", database.getData("key2"));
    }

    public Data getData(String key) {
        return cache.get(key);
    }
}
定时预热

系统定期自动加载常用数据到缓存中,确保缓存中的数据是最新的。

java 复制代码
public class CacheService {
    private final Map<String, Data> cache = new ConcurrentHashMap<>();
    private final ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(1);
    private final Database database = new Database();
    private final long CACHE_REFRESH_INTERVAL = 30; // 每 30 秒预热一次

    public CacheService() {
        // 启动定时预热任务
        scheduler.scheduleAtFixedRate(this::preheatCache, 0, CACHE_REFRESH_INTERVAL, TimeUnit.SECONDS);
    }

    private void preheatCache() {
        // 定时预热常用数据
        cache.put("key1", database.getData("key1"));
        cache.put("key2", database.getData("key2"));
    }

    public Data getData(String key) {
        return cache.get(key);
    }

    public void shutdown() {
        scheduler.shutdown();
    }
}

如何确定需要预先缓存哪些数据

确定需要预热哪些数据是缓存预热的关键和难点之一。选择错误的数据预热不仅浪费资源,也无法提高系统性能。以下是一些确定需要预热数据的方法。

基于历史访问数据

分析系统的历史访问日志,找出访问频率较高的数据,优先将这些数据缓存。

java 复制代码
public class CacheService {
    private final Map<String, Data> cache = new ConcurrentHashMap<>();
    private final Database database = new Database();
    private final AccessLog accessLog = new AccessLog();

    public CacheService() {
        // 启动时预热缓存
        preheatCache();
    }

    private void preheatCache() {
        for (String key : accessLog.getFrequentKeys()) {
            cache.put(key, database.getData(key));
        }
    }

    public Data getData(String key) {
        return cache.get(key);
    }
}

class Data {
    // 模拟数据类
}

class Database {
    public Data getData(String key) {
        // 从数据库获取数据
        return new Data();
    }
}

class AccessLog {
    // 模拟访问日志
    public List<String> getFrequentKeys() {
        // 返回访问频率较高的key
        return Arrays.asList("key1", "key2", "key3");
    }
}
基于业务逻辑

根据业务场景,确定哪些数据在系统启动时是必需的。例如,电子商务网站的首页商品推荐,新闻网站的头条新闻等。

java 复制代码
public class CacheService {
    private final Map<String, Data> cache = new ConcurrentHashMap<>();
    private final Database database = new Database();

    public CacheService() {
        // 启动时预热缓存
        preheatCache();
    }

    private void preheatCache() {
        // 根据业务逻辑预热数据
        cache.put("homepageData", database.getData("homepageData"));
        cache.put("popularProducts", database.getData("popularProducts"));
    }

    public Data getData(String key) {
        return cache.get(key);
    }
}

class Data {
    // 模拟数据类
}

class Database {
    public Data getData(String key) {
        // 从数据库获取数据
        return new Data();
    }
}
相关推荐
nFBD29OFC2 分钟前
Spring Cloud生态地图——注册、配置、网关、负载均衡与可观测的组合拳
spring·spring cloud·负载均衡
蒸汽求职9 分钟前
低延迟系统优化:针对金融 IT 与高频交易,如何从 CPU 缓存行(Cache Line)对齐展现硬核工程底蕴?
sql·算法·缓存·面试·职场和发展·金融·架构
吴声子夜歌9 分钟前
Node.js——zlib压缩模块
java·spring·node.js
数智化管理手记8 小时前
精益生产中的TPM管理是什么?一文破解设备零故障的密码
服务器·网络·数据库·低代码·制造·源代码管理·精益工程
翊谦9 小时前
Java Agent开发 Milvus 向量数据库安装
java·数据库·milvus
__土块__9 小时前
一次 Spring 事务传播机制源码走读:从误用 @Transactional 到理解嵌套事务的边界
spring·threadlocal·编程式事务·@transactional·事务传播·源码走读·requires_new
Java面试题总结9 小时前
Spring - Bean 生命周期
java·spring·rpc
難釋懷9 小时前
OpenResty实现Redis查询
数据库·redis·openresty
别抢我的锅包肉10 小时前
【MySQL】第四节 - 多表查询、多表关系全解析
数据库·mysql·datagrip
Database_Cool_10 小时前
OpenClaw-Observability:基于 DuckDB 构建 OpenClaw 的全链路可观测体系
数据库·阿里云·ai