本章节主要从底层源码探索Spring Boot中RabbitMQ如何进行消费,至于RabbitMQ是如何使用如何生产消息,本章不做过多介绍,感兴趣的小伙伴可以参考:从源码层级深入探索 Spring AMQP 如何在 Spring Boot 中实现 RabbitMQ 集成------生产者如何将消息发送到 RabbitMQ Exchange-CSDN博客文章浏览阅读14次。RabbitAutoConfiguration,RabbitAdmin,RabbitTemplate,declareExchanges,convertAndSendhttps://blog.csdn.net/qq_26733517/article/details/144511012?spm=1001.2014.3001.5501
1.Spring Boot中RabbitMQ如何消费
Spring Boot中RabbitMQ的消费使用非常简单,一般使用@RabbitListener或者@RabbitListener+@RabbitHandler,前者使用在方法上, 一个消费者只有一个方法,消息来了就是目标方法执行,不管参数类型跟消息类型是否匹配。后者使用在类上,一个消费者包含多个候选处理方法,根据消息的类型进行选择执行。具体使用如下图:
data:image/s3,"s3://crabby-images/924f3/924f35b76b6b519bd44a15d4bf4e7020f23bdba7" alt=""
那么为什么使用一个简单的注解@RabbitListener就可以实现消费呢,它的底层是如何运转实现的呢?
2.@RabbitListener注解底层如何实现消费的
其是这还是离不开Spring Boot自动装配引入的RabbitAutoConfiguration配置类,再该类中帮我们引入了一系列配置类,如下:RabbitAutoConfiguration->RabbitAnnotationDrivenConfiguration->EnableRabbitConfiguration(该配置类中某个bean对象存在@EnableRabbit注解)->
RabbitListenerConfigurationSelector(@EnableRabbit注解引入)->RabbitBootstrapConfiguration->
RabbitListenerAnnotationBeanPostProcessor、RabbitListenerEndpointRegistry。消费实现的核心就是这两个配置类,接下来我们进行逐一研究。
1.RabbitListenerAnnotationBeanPostProcessor
该配置类实现了BeanPostProcessor+SmartInitializingSingleton接口,再所有的bean对象初始化前后均会调用BeanPostProcessor接口的前置、后置方法。再所有的bean对象初始化完成后(所有的bean对象已经生成)会调用SmartInitializingSingleton接口的afterSingletonsInstantiated方法。其主要作用是用来解析@RabbitListener,具体如下:
data:image/s3,"s3://crabby-images/13795/137954c984eefca6da097860e42f425b3b61b813" alt=""
接着分别看一下processAmqpListener和processMultiMethodListeners两个方法,如下图:
data:image/s3,"s3://crabby-images/b53a9/b53a9a62e8663bb7c27a487440661b49500c8bc6" alt=""
data:image/s3,"s3://crabby-images/93c4d/93c4d0d5411df2be37a3a1cc6f79ea01d6b34187" alt=""
从上面两个图中可以看出,二者底层均是调用了processListener方法,该方法具体如下:
data:image/s3,"s3://crabby-images/88a98/88a98486de69616d61da0716c9ec0ee0b20f78c5" alt=""
接着看一下registerEndpoint方法,如下:
data:image/s3,"s3://crabby-images/52cc0/52cc00f720faaa71c1393e09519e3f2a785a20a7" alt=""
至此整个BeanPostProcessor接口的后置方法就已经完成,其作用就是将解析的@RabbitListener注解的信息封装成一个个AmqpListenerEndpointDescriptor里面封装了AbstractRabbitListenerEndpoint,然后放入到了RabbitListenerEndpointRegistrar.endpointDescriptors属性集合中。
执行完BeanPostProcessor接口后,等ioc容器的所有bean对象初始化完成后会继续调用该方法的afterSingletonsInstantiated方法,具体如下:
data:image/s3,"s3://crabby-images/1e4e9/1e4e935c61198aa60a627abaf349735ff54f5e63" alt=""
接着看一下this.registrar.afterPropertiesSet()方法,具体如下:
data:image/s3,"s3://crabby-images/660b3/660b3c9410d72764e6b6124c41aef3b8de7b77be" alt=""
接着看一下registerListenerContainer方法,具体如下:
data:image/s3,"s3://crabby-images/06b6a/06b6aa0f7b24e9f5789c70d6d3a1cb94329ca115" alt=""
至此SmartInitializingSingleton接口的方法afterSingletonsInstantiated,将每一个@RabbitListener注解的信息封装成了SimpleMessageListenerContainer,然后封装到了RabbitListenerEndpointRegistry中的listenerContainers属性集合中,该类是不是有点眼熟,就是我们开始时说的两个核心类之一,接下来我们便研究一下它。
2.RabbitListenerEndpointRegistry
RabbitListenerEndpointRegistry实现了lifecyle接口,再spring容器刷新的最后一步会调用该类的start方法,处理上面解析的@RabbitListener注解信息,具体如下:
data:image/s3,"s3://crabby-images/7859a/7859a183d590bab2960670970b67b8fdf9156188" alt=""
接着看一下startIfNecessary方法,调用每一个MessageListenerContainer(SimpleMessageListenerContainer)的start方法,如下:
data:image/s3,"s3://crabby-images/e9d0d/e9d0d409003713193dd019a728395397c9745bff" alt=""
调用的为AbstractMessageListenerContainer(父类)中的start方法,如下:
data:image/s3,"s3://crabby-images/97280/97280ed457132e07ae74718e8b9475d0622392d2" alt=""
紧接着看一下SimpleMessageListenerContainer中的doStart()方法,如下:
data:image/s3,"s3://crabby-images/920e7/920e72c8c4703d6045adff22836286dff64aa38a" alt=""
接着看一下AsyncMessageProcessingConsumer的run() ,如下:
data:image/s3,"s3://crabby-images/88ae4/88ae4adf93fc073535b0e7294c12b15ffd6351b9" alt=""
初始化的过程此处就不再介绍了,主要看一下事件如何处理的,即mainLoop方法,具体如下:
data:image/s3,"s3://crabby-images/ef571/ef57147e458bcc51ca6a98850e2c86ced17fb1da" alt=""
receiveAndExecute方法底层调用了doReceiveAndExecute方法,具体如下:
data:image/s3,"s3://crabby-images/06a73/06a7314168f3c901f7d53fc3da1fa26ffc933bd2" alt=""
目标方法的执行涉及的东西比较多,包括处理在@RabbitListener注解中使用errorHandler和returnExceptions等属性,比较复杂,本篇暂不做过多介绍。我们看一下Spring Boot的假的'自动确认'吧,即commitIfNecessary方法,如下:
data:image/s3,"s3://crabby-images/426df/426df99a2369f02fa863cfbb70b0e4f5caa1165e" alt=""
所以说Spring Boot提供所谓自动确认并不是真正的自动确认,只是Spring Boot再执行完目标方法后帮我们进行了确认,所以对开发者来说感知上是自动确认。
如果您希望更深入地学习SpringBoot源码,我强烈推荐您访问以下项目链接:https://gitee.com/chengyadong555/spring-boot.git。在这个项目中,您将发现对SpringBoot源码的逐行分析,作者不仅提供了丰富的注释,还融入了自己独到的理解和见解。