基于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)。

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

相关推荐
程序员飞哥2 分钟前
别再说“对接接口没技术含量了”,这才是高手的打开方式!
后端·程序员
DokiDoki之父6 分钟前
Spring—容器
java·后端·spring
摇滚侠19 分钟前
Spring Boot 3零基础教程,WEB 开发 国际化 Spring Boot + Thymeleaf 笔记45
spring boot·笔记·后端
间彧21 分钟前
Java AQS详解与项目实战
后端
golang学习记28 分钟前
性能飙升4倍,苹果刚发布的M5给人看呆了
人工智能·后端
程序员爱钓鱼1 小时前
Python编程实战 · 基础入门篇 | 类型转换与输入输出
后端·python
程序员爱钓鱼1 小时前
Python编程实战 · 基础入门篇 | 运算符详解
后端·python·编程语言
xiezhr1 小时前
见过哪些醍醐灌顶的Java代码:从"卧槽"到"原来如此"的顿悟
java·后端·设计模式
canonical_entropy1 小时前
Nop平台架构白皮书:一个基于广义可逆计算理论的软件构造体系评估
后端·低代码·领域驱动设计
IT_陈寒1 小时前
SpringBoot 3.2新特性盘点:这5个隐藏功能让你的开发效率翻倍 🚀
前端·人工智能·后端