基于Redis 发布订阅实现一个轻量级本地缓存刷新

基于 Redis 的发布/订阅机制,我们可以很方便地实现一个"刷新本地缓存"的工具,适用于多个实例之间同步缓存刷新的场景。

以下是一个完整可落地的实现方案,基于 SpringBoot 和 RedisTemplate 实现:


✅ 第一步:配置 Redis 发布订阅

1. 配置 RedisMessageListenerContainer(订阅容器)

java 复制代码
@Configuration
public class RedisPubSubConfig {

    @Bean
    public RedisMessageListenerContainer container(
            RedisConnectionFactory connectionFactory,
            MessageListenerAdapter listenerAdapter) {
        RedisMessageListenerContainer container = new RedisMessageListenerContainer();
        container.setConnectionFactory(connectionFactory);
        container.addMessageListener(listenerAdapter, new ChannelTopic("cache:flush"));
        return container;
    }

    @Bean
    public MessageListenerAdapter listenerAdapter(CacheFlushSubscriber subscriber) {
        return new MessageListenerAdapter(subscriber, "onMessage");
    }
}

2. 订阅者:监听缓存刷新消息

java 复制代码
@Component
public class CacheFlushSubscriber {

    @Autowired
    private LocalCacheManager localCacheManager;

    public void onMessage(String message, String channel) {
        if ("flush:all".equals(message)) {
            localCacheManager.refreshAll();
        } else if (message.startsWith("flush:")) {
            String key = message.substring(6);
            localCacheManager.refreshByKey(key);
        }
    }
}

3. 发布者:发送刷新指令

java 复制代码
@Component
public class CacheFlushPublisher {

    @Autowired
    private RedisTemplate<String, String> redisTemplate;

    public void publishFlushAll() {
        redisTemplate.convertAndSend("cache:flush", "flush:all");
    }

    public void publishFlushByKey(String key) {
        redisTemplate.convertAndSend("cache:flush", "flush:" + key);
    }
}

✅ 第二步:本地缓存管理器(示例)

java 复制代码
@Component
public class LocalCacheManager {

    private final Map<String, Object> cache = new ConcurrentHashMap<>();

    // 模拟从数据库加载
    public void refreshAll() {
        cache.clear();
        cache.put("user:1", "Alice");
        cache.put("user:2", "Bob");
        System.out.println("本地缓存已刷新全部");
    }

    public void refreshByKey(String key) {
        cache.remove(key);
        cache.put(key, "refreshed-value-for-" + key);
        System.out.println("本地缓存已刷新 key: " + key);
    }

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

✅ 第三步:测试接口(可选)

java 复制代码
@RestController
@RequestMapping("/cache")
public class CacheController {

    @Autowired
    private CacheFlushPublisher publisher;

    @GetMapping("/flush/all")
    public String flushAll() {
        publisher.publishFlushAll();
        return "已发送刷新全部缓存指令";
    }

    @GetMapping("/flush/{key}")
    public String flushByKey(@PathVariable String key) {
        publisher.publishFlushByKey(key);
        return "已发送刷新指定缓存指令:" + key;
    }

    @GetMapping("/get/{key}")
    public Object get(@PathVariable String key) {
        return localCacheManager.get(key);
    }
}

✅ 使用方式

  • 访问 /cache/flush/all → 所有实例本地缓存刷新
  • 访问 /cache/flush/user:1 → 仅刷新 key 为 user:1 的缓存
  • 访问 /cache/get/user:1 → 获取本地缓存值

✅ 注意事项

  • 发布/订阅消息不会持久化,如果实例未启动,将不会收到历史消息
  • 适用于轻量级广播,不适合高可靠性消息传递。
  • 若需强一致性,建议使用消息队列(如 Kafka、RabbitMQ)。

如需扩展支持多个缓存区域、异步刷新、延迟刷新等功能,可在此基础上继续抽象封装。

相关推荐
码事漫谈31 分钟前
大模型输出的“隐性结构塌缩”问题及对策
前端·后端
小江的记录本1 小时前
【网络安全】《网络安全常见攻击与防御》(附:《六大攻击核心特性横向对比表》)
java·网络·人工智能·后端·python·安全·web安全
努力的小雨1 小时前
龙虾量化实战法(QClaw)
后端
橙露2 小时前
SpringBoot 整合 MinIO:分布式文件存储上传下载
spring boot·分布式·后端
2401_895521343 小时前
【Spring Security系列】Spring Security 过滤器详解与基于JDBC的认证实现
java·后端·spring
小码哥_常3 小时前
大文件上传不再卡顿:Spring Boot 分片上传、断点续传与进度条实现全解析
后端
_Evan_Yao4 小时前
RAG中的“Chunk”艺术:我试过10种切分策略后总结的结论
java·人工智能·后端·python·软件工程
今天你TLE了吗4 小时前
LLM到Agent&RAG——AI概念概述 第二章:提示词
人工智能·笔记·后端·学习
IT_陈寒5 小时前
Vue的响应式把我坑惨了,原来问题出在这
前端·人工智能·后端
shark22222225 小时前
能懂!基于Springboot的用户增删查改(三层设计模式)
spring boot·后端·设计模式