当前位置: 首页 > news >正文

【2025最新Java面试八股】如何在Spring启动过程中做缓存预热?

在 Spring 应用启动过程中进行缓存预热(Cache Preloading)是一种优化手段,目的是在系统正式对外提供服务前,提前加载高频访问的数据到缓存(如 Redis、Caffeine 等),避免用户首次请求时因缓存未命中(Cache Miss)导致性能下降。以下是详细的实现方案和最佳实践:

1. 缓存预热的核心思路

  • 目标:在 Spring 容器初始化完成后,主动加载热点数据到缓存。

  • 适用场景

    • 高频访问的静态数据(如配置表、城市列表)。

    • 计算成本高的数据(如排行榜、聚合统计结果)。

  • 关键时机:确保预热在应用完全启动后执行,且不影响正常服务。


2. 实现方案

方案1:使用 @PostConstruct 注解或者实现 InitializingBean 接口,
 实现 InitializingBean 接口,可以重写afterPropertiesSet 方法中执行缓存预热的逻辑。这样,Spring 在初始化 Bean 时会调用 afterPropertiesSet 方法。

在 Spring Bean 初始化完成后立即执行预热逻辑:

@Service
public class CacheWarmUpService {@Autowiredprivate UserService userService; // 依赖的业务服务@Autowiredprivate CacheManager cacheManager; // Spring 缓存管理器@PostConstruct  // 在 Bean 初始化后执行public void warmUpCache() {List<User> hotUsers = userService.getTopActiveUsers(100); // 加载热点数据hotUsers.forEach(user -> cacheManager.getCache("userCache").put(user.getId(), user));}
}

import org.springframework.beans.factory.InitializingBean;

import org.springframework.stereotype.Component;

@Component

public class CachePreloader implements InitializingBean {

                 @Override

                public void afterPropertiesSet() throws Exception {

                        // 执行缓存预热逻辑

                        // ...

                }

}

优点:简单直接,适合小规模预热。
缺点:可能阻塞 Spring 启动流程,需控制预热时间。


方案2:实现 ApplicationListener 监听上下文就绪事件

ApplicationReadyEvent 是 Spring Boot 框架中的一个事件类,它表示应用程序已经准备好接收请求,即应用程序已启动且上下文已刷新。这个事件是在 ApplicationContext 被初始化和刷新,并且应用程序已经准备好处理请求时触发的。

基于ApplicationReadyEvent,我们可以在应用程序完全启动并处于可用状态后执行一些初始化逻辑。使用 @EventListener 注解或实现 ApplicationListener 接口来监听这个事件。

在 Spring 上下文完全初始化后触发预热:

@Component
public class CacheWarmUpListener implements  ApplicationListener<ContextRefreshedEvent> {
//通过实现ApplicationListener<ContextRefreshedEvent>接口
//也可以通过@EventListener实现@Overridepublic void onApplicationEvent(ContextRefreshedEvent event) {if (event.getApplicationContext().getParent() == null) { // 避免子容器重复执行// 执行预热逻辑warmUpRedisCache();warmUpLocalCache();}}private void warmUpRedisCache() {// 例如:预加载商品数据到 Redis// ...}
}

优点:确保所有 Bean 已就绪,适合复杂预热逻辑。
缺点:需注意避免重复执行(如 Web 应用可能有父子容器)。


方案3:使用 CommandLineRunner 或 ApplicationRunner

Spring Boot 提供的启动后扩展点:

@Component
@Order(1) // 控制执行顺序
public class CacheWarmUpRunner implements CommandLineRunner {@Autowiredprivate ProductService productService;@Overridepublic void run(String... args) {List<Product> hotProducts = productService.getHotProducts();// 写入缓存(如 Caffeine、Redis)}
}

优点:与 Spring Boot 生命周期无缝集成,支持多任务顺序控制。
缺点:仅适用于 Spring Boot 项目。


方案4:异步预热(推荐)

避免阻塞主线程,使用 @Async 异步执行:

@Service
public class AsyncCacheWarmUpService {@Async  // 需启用 @EnableAsyncpublic void warmUpCacheAsync() {// 耗时预热逻辑(如全量加载数据库数据到缓存)}
}// 通过监听器或 Runner 触发异步任务
@Component
public class CacheWarmUpTrigger implements ApplicationListener<ContextRefreshedEvent> {@Autowiredprivate AsyncCacheWarmUpService asyncService;@Overridepublic void onApplicationEvent(ContextRefreshedEvent event) {asyncService.warmUpCacheAsync();}
}

优点:不阻塞应用启动,适合大数据量预热。
注意:需配置线程池(通过 ThreadPoolTaskExecutor)。


3. 结合具体缓存框架

3.1 Redis 预热
public void warmUpRedis() {StringRedisTemplate redisTemplate = ...;List<City> cities = cityRepository.findAll();cities.forEach(city -> redisTemplate.opsForValue().set("city:" + city.getId(), city.getName()));
}
3.2 Caffeine 本地缓存预热
@Bean
public Cache<String, Product> productCache() {return Caffeine.newBuilder().maximumSize(1000).build(cache -> {// 启动时自动加载数据return productService.getProductById(cache.key());});
}

4. 最佳实践

  1. 分批次加载:避免单次加载数据量过大导致 OOM 或超时。

    List<User> users = userService.getAllUsers();
    int batchSize = 100;
    for (int i = 0; i < users.size(); i += batchSize) {List<User> batch = users.subList(i, Math.min(i + batchSize, users.size()));batch.forEach(user -> cache.put(user.getId(), user));
    }
  2. 动态调整预热策略:通过配置文件控制是否启用预热或选择预热数据集。

    cache:warm-up:enabled: truedata-sets: top_users,hot_products
  3. 监控与日志:记录预热耗时和数据量,便于优化。

    @Slf4j
    public class CacheWarmUpRunner implements CommandLineRunner {@Overridepublic void run(String... args) {long start = System.currentTimeMillis();// 预热逻辑log.info("Cache warm-up completed in {} ms", System.currentTimeMillis() - start);}
    }

5. 避免的坑

  • 循环依赖:预热 Bean 不要依赖其他未初始化的 Bean。

  • 事务问题确保预热方法内数据库操作已提交(可加 @Transactional)。

  • 分布式环境在集群中仅由一个节点执行预热(通过分布式锁控制)。


总结

方案适用场景是否阻塞启动
@PostConstruct简单、小数据量预热
ApplicationListener需要完整上下文
CommandLineRunnerSpring Boot 项目,需控制顺序
异步预热(推荐)大数据量或耗时任务

选择合适方案后,结合具体缓存框架(Redis/Caffeine)和业务需求,可显著提升系统启动后的缓存命中率。

相关文章:

  • kafka 中消费者 groupId 是什么
  • [python] 基于WatchDog库实现文件系统监控
  • Seaborn模块练习题
  • GCC 内建函数汇编展开详解
  • 【数据挖掘】时间序列预测-时间序列预测策略
  • 脏读、幻读、可重复读
  • 反序列化漏洞2
  • 数据结构(七)---链式栈
  • 力扣HOT100之链表:23. 合并 K 个升序链表
  • ComfyUI for Windwos与 Stable Diffusion WebUI 模型共享修复
  • JavaScript 中 undefined 和 not defined 的区别
  • 【深度学习】多头注意力机制的实现|pytorch
  • 生物医学AI的特种算力需求:冷冻电镜数据处理中的GPU加速方案
  • GIS开发笔记(16)解决基于osg和osgearth三维地图上添加placeNode图标点击不易拾取的问题
  • UML 活动图详解之小轿车启动活动图分析
  • Dev控件RadioGroup 如何设置一排有N个显示或分为几行
  • 在Linux中,使用IO标准库进行读写操作。
  • 塔能合作模式:解锁工厂能耗精准节能新路径
  • Flutter 泛型 泛型方法 泛型类 泛型接口
  • 《原神/星穹铁道私服怎么建?内网穿透+本地调试完整指南》
  • 国家发改委答澎湃:力争6月底前下达2025年两重建设和中央预算内投资全部项目清单
  • 央媒关注给保洁人员设休息室:让每一份踏实奋斗得到尊重呵护
  • 中国海警局新闻发言人就菲律宾非法登临铁线礁发表谈话
  • 只在上海!德国剧团新作亚洲首演:一张古典与流行的声音网络
  • 建投读书会·东西汇流|东西方戏剧在上海的相逢、交锋与融合
  • 特斯拉季度利润暴跌71%,马斯克下月开始大幅减少为政府工作时间