Spring指南之使用 Redis 进行消息传递(Spring Data Redis/以及自定义注解实现)

本文将指导你使用Spring Data Redis构建一个消息发布应用,通过POJO实现接收器,并演示如何发送和订阅Redis消息。学习如何配置Redis连接、监听器与模板,以及使用Redis客户端工具进行验证。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

请参见官方文档Spring指南之Messaging with Redis
参见代码地址
请参见Redis菜鸟教程之Redis 发布订阅
请参见请参见Spring Data Redis官方文档之Redis 消息传递(发布/订阅)

一.简介

本指南将向您介绍如何使用Spring Data Redis发布和订阅用Redis发送的消息。

二.你将创造什么(What You Will Build)

1.您将构建一个使用StringRedisTemplate发布字符串消息的应用程序,并通过使用MessageListenerAdapter为该消息提供一个POJO订阅

2.使用Spring Data Redis作为发布消息的手段听起来可能有些奇怪,但是,正如您将发现的,Redis不仅提供了NoSQL数据存储,还提供了消息传递系统

三.创建项目

创建项目过程请参见Spring入门指南之创建多模块项目

1.Dependencies选择Spring Data Redis

四.创建Redis消息接收器

1.在任何基于消息传递的应用程序中,都存在消息发布者和消息接收者。要创建消息接收器,用一个响应消息的方法实现一个接收器。

//一个自定义的消息接收器
public class MyReceiver1 {
    private static final Logger LOGGER = LoggerFactory.getLogger(MyReceiver1.class);
    private AtomicInteger counter = new AtomicInteger();
    
    public void receiveMessage(String message) {
		counter.incrementAndGet();
        LOGGER.info("Received <" + message + ">----counter=" + getCount());
    }

    public int getCount() {
        return counter.get();
    }
}

五.注册监听器并发送消息

1.Spring Data Redis提供了所有你需要用Redis发送和接收消息的组件。具体来说,你需要配置:

  • A connection factory 连接工厂
  • A message listener container 消息监听器容器
  • A Redis template Redis模板

2.(1)您将使用Redis template发送消息,并将接收者(MyReceiver1)注册到消息监听器容器中,以便它接收消息。连接工厂(RedisConnectionFactory)驱动 template 和消息监听器容器,让它们连接到Redis服务器。这个例子使用了Spring Boot默认的RedisConnectionFactory,这是一个基于Jedis Redis库的JedisConnectionFactory实例。连接工厂被注入到消息监听器容器和Redis template中,如下例所示。
(2)listenerAdapter方法中定义的bean被注册为容器中定义的消息监听器容器中的消息监听器,并将监听有关 chat 主题的消息。
因为MyReceiver1类是一个POJO,所以需要将其包装在实现MessageListener接口的消息监听器适配器(MessageListenerAdapter)中(addMessageListener()需要这个接口)。消息监听器适配器也配置为在消息到达时调用MyReceiver1上的receiveMessage()方法。
(3)监听消息只需要连接工厂和消息监听器容器bean。要发送消息,还需要Redis模板。在这里,它是一个配置为StringRedisTemplate的bean,RedisTemplate的一个实现,其中键和值都是字符串实例。

@Configuration
public class RedisMessageListener {

    @Bean
    MyReceiver1 myReceiver1() {
        return new MyReceiver1();
    }

    //使用消息监听器容器注册Receiver,以便它将接收消息(监听receiveMessage方法)
    @Bean
    MessageListenerAdapter listenerAdapter(MyReceiver1 myReceiver1) {
        return new MessageListenerAdapter(myReceiver1, "receiveMessage");
    }

    //您将使用Redis模板发送消息
    //连接工厂被注入到 Redis模板中
    @Bean
    StringRedisTemplate template(RedisConnectionFactory connectionFactory) {
        return new StringRedisTemplate(connectionFactory);
    }

    //使用了Spring Boot默认的RedisConnectionFactory,这是一个基于Jedis Redis库的JedisConnectionFactory实例。
    //连接工厂被注入到消息侦听器容器中
    @Bean
    RedisMessageListenerContainer container(RedisConnectionFactory connectionFactory,
                                            MessageListenerAdapter listenerAdapter) {

        RedisMessageListenerContainer container = new RedisMessageListenerContainer();
        container.setConnectionFactory(connectionFactory);
        container.addMessageListener(listenerAdapter, new PatternTopic("chat"));

        return container;
    }
}

3.在配置文件中添加redis相关配置

spring.redis.host=localhost
spring.redis.port=6379
spring.redis.password=xxxx

4.在启动类中创建 Spring 应用程序上下文(Spring application context)。应用程序上下文启动消息监听器容器,消息监听器容器bean开始监听消息。从应用程序上下文中检索StringRedisTemplate bean,并使用它从Redis发送关于 chat 主题的消息。运行启动类开始测试。

@SpringBootApplication
public class GsMessagingRedisApplication {
    public static void main(String[] args) throws InterruptedException {
        ApplicationContext ac = SpringApplication.run(GsMessagingRedisApplication.class, args);
        StringRedisTemplate template = ac.getBean(StringRedisTemplate.class);
        MyReceiver1 receiver = ac.getBean(MyReceiver1.class);
        while (true) {
            template.convertAndSend("chat", "Hello World!");
            Thread.sleep(500L);
        }
    }
}

在这里插入图片描述

六.使用redis客户端工具订阅消息

在这里插入图片描述

七.问题

实际上我们不需要再去配置StringRedisTemplate bean–>因为Spring boot自动配置了
在这里插入图片描述

八.自己扩展自定义注解实现(模仿@RabbitListener)

1.创建一个注解类RedisListener

/**
 * 模仿@RabbitListener
 */
@Target({ElementType.METHOD})   //作用于方法上
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface RedisListener {
	//消息订阅的主题
    String topic() default "";
}

2.创建一个自定义消息监听器

//一个自定义的消息接收器
@Component
public class MyReceiver2 {
    private static final Logger LOGGER = LoggerFactory.getLogger(MyReceiver2.class);

    @RedisListener(topic = "chat2")
    public void receiveMessage(String message) {
        LOGGER.info("MyReceiver2 Received <" + message + ">");
    }
}

3.编写一个初始化的类中完成我们之前注册监听器的步骤!!!

请请参看spring-boot-starter-amqp启动器的RabbitListenerAnnotationBeanPostProcessor类和RabbitListenerEndpointRegistrar类和RabbitListenerEndpointRegistry类源码

@Component
public class RedisListenerAnnotationBeanPostProcessor implements BeanPostProcessor {
    @Autowired
    private ApplicationContext applicationContext;

    /**
     * 注册MessageListenerAdapter bean--->等价于RedisMessageListener类的listenerAdapter(xx)方法
     */
    @Nullable
    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        Method[] declaredMethods = bean.getClass().getDeclaredMethods();
        for (Method declaredMethod : declaredMethods) {
            if (declaredMethod.isAnnotationPresent(RedisListener.class)) {
                String topic = declaredMethod.getAnnotation(RedisListener.class).topic();
                DefaultListableBeanFactory defaultListableBeanFactory = (DefaultListableBeanFactory) applicationContext.getAutowireCapableBeanFactory();
                //多个MessageListenerAdapter的beanName加topic区分
                String messageListenerAdapterBeanName = String.format("%s_%s", topic, MessageListenerAdapter.class.getName());
                //注册MessageListenerAdapter bean
                MessageListenerAdapter messageListenerAdapter = new MessageListenerAdapter(bean, declaredMethod.getName());
                //这里需要手动调用一下初始化回调,不然messageListenerAdapter.onMessage会报错空指针异常
                messageListenerAdapter.afterPropertiesSet();
                defaultListableBeanFactory.registerSingleton(messageListenerAdapterBeanName, messageListenerAdapter);
            }
        }
        return bean;
    }

    @Nullable
    @Override
    public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        return bean;
    }

}
    //向RedisMessageListenerContainer添加MessageListenerAdapte
    @Bean
    CommandLineRunner init() {
        return new CommandLineRunner() {
            @Override
            public void run(String... args) throws Exception {
                RedisMessageListenerContainer redisMessageListenerContainer = applicationContext.getBean(RedisMessageListenerContainer.class);
                Map<String, MessageListenerAdapter> beansOfType = applicationContext.getBeansOfType(MessageListenerAdapter.class);
                for (String messageListenerAdapterBeanName : beansOfType.keySet()) {
                    //排除一下MyReceiver1
                    if(messageListenerAdapterBeanName.contains("_")){
                        String topic = messageListenerAdapterBeanName.substring(0, messageListenerAdapterBeanName.indexOf("_"));
                        redisMessageListenerContainer.addMessageListener((MessageListenerAdapter) applicationContext.getBean(messageListenerAdapterBeanName), new PatternTopic(topic));
                    }
                }
            }
        };
    }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值