spring boot 启动的时候将数据库里的分类信息写入redis

在 Spring Boot 启动时将数据库分类信息写入 Redis,推荐使用 ApplicationRunnerCommandLineRunner。它们会在 Spring 容器完全初始化、所有 Bean 就绪后执行,非常适合做缓存预热。

下面提供一份生产可用的完整方案:

📦 1. 核心实现代码

复制代码
复制代码
import com.fasterxml.jackson.databind.ObjectMapper;
import lombok.RequiredArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.boot.ApplicationArguments;
import org.springframework.boot.ApplicationRunner;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.stereotype.Component;

import java.util.List;

@Slf4j
@Component
@RequiredArgsConstructor
public class CategoryCacheWarmer implements ApplicationRunner {

    private final CategoryRepository categoryRepository;
    private final StringRedisTemplate stringRedisTemplate;
    private final ObjectMapper objectMapper;

    private static final String CATEGORY_CACHE_KEY = "app:categories";

    @Override
    public void run(ApplicationArguments args) {
        try {
            log.info("🚀 开始预热分类数据到 Redis...");
            
            List<Category> categories = categoryRepository.findAll();
            if (categories.isEmpty()) {
                log.warn("⚠️ 数据库中无分类数据,跳过 Redis 缓存初始化");
                return;
            }

            // 转为 JSON 字符串写入 Redis
            String json = objectMapper.writeValueAsString(categories);
            stringRedisTemplate.opsForValue().set(CATEGORY_CACHE_KEY, json);
            
            log.info("✅ 成功将 {} 条分类数据写入 Redis,Key: {}", categories.size(), CATEGORY_CACHE_KEY);
        } catch (Exception e) {
            // ⚠️ 建议不要直接抛出异常阻断应用启动,除非该缓存是强依赖
            log.error("❌ 初始化分类缓存失败,请检查数据库或 Redis 连接", e);
            // 如需阻断启动可取消注释: throw new RuntimeException("Category cache init failed", e);
        }
    }
}

⚙️ 2. 关键配置说明

Spring Boot 默认配置的 RedisTemplate 使用 JDK 序列化,会导致 Redis 中出现乱码。强烈建议使用 StringRedisTemplate + JSON 序列化

复制代码
复制代码
// Spring Boot 已自动配置 StringRedisTemplate,通常无需额外配置
// 若需自定义 Jackson 序列化行为(如忽略 null、日期格式等),可注册 ObjectMapper Bean:
@Configuration
public class RedisConfig {
    @Bean
    public ObjectMapper redisObjectMapper() {
        ObjectMapper mapper = new ObjectMapper();
        mapper.setSerializationInclusion(JsonInclude.Include.NON_NULL); // 忽略 null
        return mapper;
    }
}

🛡️ 3. 生产环境注意事项

问题 建议方案
Redis 宕机导致启动失败 使用 try-catch 降级记录日志,不阻断启动(如上代码)
每次启动都覆盖缓存? 启动时覆盖是合理的。若需保留运行期缓存,可先 hasKey() 判断,或使用版本号 Key:app:categories:v2
数据量大导致 OOM 或阻塞 分类数据通常较小。若超千条,建议分页查询或使用 Redis Pipeline 批量写入
运行期数据更新如何同步? 启动预热仅解决"冷启动"。CRUD 操作需通过 Cache Evict/Update 或监听 Binlog/MQ 保持缓存一致性
多实例部署重复写入 Redis 写入是幂等的,多实例同时执行无副作用。若需控制执行节点,可加分布式锁或配置 @ConditionalOnProperty

🔍 4. 为什么不用 @PostConstruct@EventListener

  • @PostConstruct:在 Bean 初始化后执行,但此时 Spring 上下文可能未完全就绪,DB/Redis 连接可能未建立。
  • @EventListener(ApplicationReadyEvent.class):功能等价于 ApplicationRunner,但 ApplicationRunner 接口更语义化,且能方便传入启动参数。

📌 5. 扩展:按需开关预热

可通过配置文件控制是否开启:

复制代码
复制代码
app:
  cache:
    warmup:
      enabled: true
      category: true
复制代码
复制代码
@Component
@RequiredArgsConstructor
@ConditionalOnProperty(prefix = "app.cache.warmup", name = "enabled", havingValue = "true", matchIfMissing = true)
public class CategoryCacheWarmer implements ApplicationRunner {
    // ... 同上文
}

💡 最佳实践总结 :启动预热适合读多写少、变更频率低的基础数据(如分类、字典、配置)。对于高频变更数据,应采用"缓存穿透防护 + 主动更新/过期策略"代替启动加载。

相关推荐
iAm_Ike8 小时前
Go 中自定义类型与基础类型间的显式类型转换详解
jvm·数据库·python
iuvtsrt8 小时前
Golang怎么实现方法集与接口的匹配_Golang如何理解值类型和指针类型实现接口的区别【详解】
jvm·数据库·python
難釋懷9 小时前
Redis数据结构-Set结构
数据结构·redis·bootstrap
tongluowan00710 小时前
MySQL中列数量及长度
数据库·mysql
-liming-10 小时前
单片机设计_串口调试工具
数据库·单片机·mongodb
鹿角片ljp10 小时前
从告警检测到智能研判:SQL 注入研判模型的设计与实践
数据库·sql
秋漓11 小时前
Redis学习笔记
redis
小新同学^O^11 小时前
简单学习 --> Spring事务
数据库·学习·spring
前进的李工11 小时前
MySQL慢查询日志优化实战
数据库·mysql·性能优化
KaMeidebaby12 小时前
卡梅德生物技术快报|禽类成纤维细胞 FISH 实验:鸟类性别染色体基因定位技术实现与数据验证
前端·数据库·其他·百度·新浪微博