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

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

相关推荐
q***o37624 分钟前
Spring Boot环境配置
java·spring boot·后端
hhzz25 分钟前
Springboot项目中使用POI操作Excel(详细教程系列3/3)
spring boot·后端·excel·poi·easypoi
TaiKuLaHa39 分钟前
Spring Bean的生命周期
java·后端·spring
JavaGuide1 小时前
SpringBoot 官宣停止维护 3.2.x~3.4.x!
java·后端
Victor3562 小时前
Hibernate(39)Hibernate中如何使用拦截器?
后端
Victor3562 小时前
Hibernate(40)Hibernate的命名策略是什么?
后端
Knight_AL2 小时前
Spring 事务管理:为什么内部方法调用事务不生效以及如何解决
java·后端·spring
bcbnb2 小时前
iOS代码混淆技术深度实践:从基础到高级全面解析
后端
加洛斯3 小时前
SpringSecurity入门篇(2):替换登录页与config配置
前端·后端