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

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

相关推荐
Coder_Boy_38 分钟前
基于SpringAI的在线考试系统-考试系统开发流程案例
java·数据库·人工智能·spring boot·后端
掘金者阿豪2 小时前
关系数据库迁移的“暗礁”:金仓数据库如何规避数据完整性与一致性风险
后端
ServBay2 小时前
一个下午,一台电脑,终结你 90% 的 Symfony 重复劳动
后端·php·symfony
sino爱学习2 小时前
高性能线程池实践:Dubbo EagerThreadPool 设计与应用
java·后端
颜酱2 小时前
从二叉树到衍生结构:5种高频树结构原理+解析
javascript·后端·算法
掘金者阿豪2 小时前
UUID的隐形成本:一个让数据库“慢下来”的陷阱
后端
用户084465256373 小时前
Docker 部署 MongoDB Atlas 到服务端
后端
Anita_Sun3 小时前
一看就懂的 Haskell 教程 - 类型推断机制
后端·haskell
Anita_Sun3 小时前
一看就懂的 Haskell 教程 - 类型签名
后端·haskell
七八星天3 小时前
C#代码设计与设计模式
后端