Java SpringBoot集成RabbitMq实战和总结

目录

交换器、队列、绑定的声明

关于消息序列化

同一个队列多消费类型

注解将消息和消息头注入消费者方法

关于消费者确认

关于发送者确认模式

消费消息、死信队列和RetryTemplate

RPC模式的消息(不常用)

关于消费模型

关于RabbitMq客户端的线程模型

在公司里一直在用RabbitMQ,由于api已经封装的很简单,关于RabbitMQ本身还有封装的实现没有了解,最近在看RabbitMQ实战这本书,结合网上的一些例子和spring文档,实现了RabbitMQ和spring的集成,对着自己平时的疑惑做了一些总结。

关于RabbitMQ基础不在详细讲解(本文不适合RabbitMq零基础),RabbitMQ实战的1,2,4三章讲的非常不错。因为书中讲的都是Python和Php的例子,所以自己结合SpringBoot文档和朱小厮的博客做了一些总结,写了一些Springboot的例子。

交换器、队列、绑定的声明

SpringAMQP项目对RabbitMQ做了很好的封装,可以很方便的手动声明队列,交换器,绑定。如下:

/**? ? * 队列? ? *@return*/@Bean@Qualifier(RabbitMQConstant.PROGRAMMATICALLY_QUEUE)Queuequeue(){returnnewQueue(RabbitMQConstant.PROGRAMMATICALLY_QUEUE,false,false,true);? ? }/**? ? * 交换器? ? *@return*/@Bean@Qualifier(RabbitMQConstant.PROGRAMMATICALLY_EXCHANGE)TopicExchangeexchange(){returnnewTopicExchange(RabbitMQConstant.PROGRAMMATICALLY_EXCHANGE,false,true);? ? }/**? ? * 声明绑定关系? ? *@return*/@BeanBindingbinding(@Qualifier(RabbitMQConstant.PROGRAMMATICALLY_EXCHANGE)TopicExchange exchange,? ? ? ? ? ? ? ? ? ? @Qualifier(RabbitMQConstant.PROGRAMMATICALLY_QUEUE)Queue queue){returnBindingBuilder.bind(queue).to(exchange).with(RabbitMQConstant.PROGRAMMATICALLY_KEY);? ? }/**? ? * 声明简单的消费者,接收到的都是原始的{@linkMessage}? ? *? ? *@paramconnectionFactory? ? *? ? *@return*/@BeanSimpleMessageListenerContainersimpleContainer(ConnectionFactory connectionFactory){? ? ? ? SimpleMessageListenerContainer container =newSimpleMessageListenerContainer();? ? ? ? container.setConnectionFactory(connectionFactory);? ? ? ? container.setMessageListener(message -> log.info("simple receiver,message:{}", message));? ? ? ? container.setQueueNames(RabbitMQConstant.PROGRAMMATICALLY_QUEUE);returncontainer;? ? }

消费者和生产者都可以声明,交换器这种一般经常创建,可以手动创建。需要注意对于没有路由到队列的消息会被丢弃。

如果是Spring的话还需要声明连接:

@BeanConnectionFactoryconnectionFactory(@Value("${spring.rabbitmq.port}")intport,? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? @Value("${spring.rabbitmq.host}")String host,? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? @Value("${spring.rabbitmq.username}")String userName,? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? @Value("${spring.rabbitmq.password}")String password,? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? @Value("${spring.rabbitmq.publisher-confirms}")booleanisConfirm,? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? @Value("${spring.rabbitmq.virtual-host}")String vhost){? ? ? ? CachingConnectionFactory connectionFactory =newCachingConnectionFactory();? ? ? ? connectionFactory.setHost(host);? ? ? ? connectionFactory.setVirtualHost(vhost);? ? ? ? connectionFactory.setPort(port);? ? ? ? connectionFactory.setUsername(userName);? ? ? ? connectionFactory.setPassword(password);? ? ? ? connectionFactory.setPublisherConfirms(isConfirm);? ? }

在配置类使用@EnableRabbit的情况下,也可以基于注解进行声明,在Bean的方法上加上@RabbitListener,如下:

/**? ? * 可以直接通过注解声明交换器、绑定、队列。但是如果声明的和rabbitMq中已经存在的不一致的话? ? * 会报错便于测试,我这里都是不使用持久化,没有消费者之后自动删除? ? * {@linkRabbitListener}是可以重复的。并且声明队列绑定的key也可以有多个.? ? *? ? *@paramheaders? ? *@parammsg? ? */@RabbitListener(? ? ? ? bindings =@QueueBinding(? ? ? ? ? ? exchange =@Exchange(value = RabbitMQConstant.DEFAULT_EXCHANGE, type = ExchangeTypes.TOPIC,? ? ? ? ? ? ? ? durable = RabbitMQConstant.FALSE_CONSTANT, autoDelete = RabbitMQConstant.true_CONSTANT),? ? ? ? ? ? value =@Queue(value = RabbitMQConstant.DEFAULT_QUEUE, durable = RabbitMQConstant.FALSE_CONSTANT,? ? ? ? ? ? ? ? autoDelete = RabbitMQConstant.true_CONSTANT),? ? ? ? ? ? key = DKEY? ? ? ? ),//手动指明消费者的监听容器,默认Spring为自动生成一个SimpleMessageListenerContainercontainerFactory ="container",//指定消费者的线程数量,一个线程会打开一个Channel,一个队列上的消息只会被消费一次(不考虑消息重新入队列的情况),下面的表示至少开启5个线程,最多10个。线程的数目需要根据你的任务来决定,如果是计算密集型,线程的数目就应该少一些concurrency ="5-10")publicvoidprocess(@Headers Map<String, Object> headers, @Payload ExampleEvent msg){? ? ? ? log.info("basic consumer receive message:{headers = ["+ headers +"], msg = ["+ msg +"]}");? ? }/**? ? * {@linkQueue#ignoreDeclarationExceptions}声明队列会忽略错误不声明队列,这个消费者仍然是可用的? ? *? ? *@paramheaders? ? *@parammsg? ? */@RabbitListener(queuesToDeclare =@Queue(value = RabbitMQConstant.DEFAULT_QUEUE, ignoreDeclarationExceptions = RabbitMQConstant.true_CONSTANT))publicvoidprocess2(@Headers Map<String, Object> headers, @Payload ExampleEvent msg){? ? ? ? log.info("basic2 consumer receive message:{headers = ["+ headers +"], msg = ["+ msg +"]}");? ? }

关于消息序列化

这个比较简单,默认采用了Java序列化,我们一般使用的Json格式,所以配置了Jackson,根据自己的情况来,直接贴代码:

@BeanMessageConvertermessageConverter(){returnnewJackson2JsonMessageConverter();? ? }

同一个队列多消费类型

如果是同一个队列多个消费类型那么就需要针对每种类型提供一个消费方法,否则找不到匹配的方法会报错,如下:

@Component@Slf4j@RabbitListener(? ? bindings =@QueueBinding(? ? ? ? exchange =@Exchange(value = RabbitMQConstant.MULTIPART_HANDLE_EXCHANGE, type = ExchangeTypes.TOPIC,? ? ? ? ? ? durable = RabbitMQConstant.FALSE_CONSTANT, autoDelete = RabbitMQConstant.true_CONSTANT),? ? ? ? value =@Queue(value = RabbitMQConstant.MULTIPART_HANDLE_QUEUE, durable = RabbitMQConstant.FALSE_CONSTANT,? ? ? ? ? ? autoDelete = RabbitMQConstant.true_CONSTANT),? ? ? ? key = RabbitMQConstant.MULTIPART_HANDLE_KEY? ? ))@Profile(SpringConstant.MULTIPART_PROFILE)publicclassMultipartConsumer{/**? ? * RabbitHandler用于有多个方法时但是参数类型不能一样,否则会报错? ? *? ? *@parammsg? ? */@RabbitHandlerpublicvoidprocess(ExampleEvent msg){? ? ? ? log.info("param:{msg = ["+ msg +"]} info:");? ? }@RabbitHandlerpublicvoidprocessMessage2(ExampleEvent2 msg){? ? ? ? log.info("param:{msg2 = ["+ msg +"]} info:");? ? }/**? ? * 下面的多个消费者,消费的类型不一样没事,不会被调用,但是如果缺了相应消息的处理Handler则会报错? ? *? ? *@parammsg? ? */@RabbitHandlerpublicvoidprocessMessage3(ExampleEvent3 msg){? ? ? ? log.info("param:{msg3 = ["+ msg +"]} info:");? ? }}

注解将消息和消息头注入消费者方法

在上面也看到了@Payload等注解用于注入消息。这些注解有:

@Header 注入消息头的单个属性

@Payload 注入消息体到一个JavaBean中

@Headers 注入所有消息头到一个Map中

这里有一点主要注意,如果是com.rabbitmq.client.Channel,org.springframework.amqp.core.Message和org.springframework.messaging.Message这些类型,可以不加注解,直接可以注入。

如果不是这些类型,那么不加注解的参数将会被当做消息体。不能多于一个消息体。如下方法ExampleEvent就是默认的消息体:

publicvoidprocess2(@Headers Map<String, Object> headers,ExampleEvent msg);

关于消费者确认

RabbitMq消费者可以选择手动和自动确认两种模式,如果是自动,消息已到达队列,RabbitMq对无脑的将消息抛给消费者,一旦发送成功,他会认为消费者已经成功接收,在RabbitMq内部就把消息给删除了。另外一种就是手动模式,手动模式需要消费者对每条消息进行确认(也可以批量确认),RabbitMq发送完消息之后,会进入到一个待确认(unacked)的队列,如下图红框部分:

如果消费者发送了ack,RabbitMq将会把这条消息从待确认中删除。如果是nack并且指明不要重新入队列,那么该消息也会删除。但是如果是nack且指明了重新入队列那么这条消息将会入队列,然后重新发送给消费者,被重新投递的消息消息头amqp_redelivered属性会被设置成true,客户端可以依靠这点来判断消息是否被确认,可以好好利用这一点,如果每次都重新回队列会导致同一消息不停的被发送和拒绝。消费者在确认消息之前和RabbitMq失去了连接那么消息也会被重新投递。所以手动确认模式很大程度上提高可靠性。自动模式的消息可以提高吞吐量。

spring手动确认消息需要将SimpleRabbitListenerContainerFactory设置为手动模式:

? ? ? ? simpleRabbitListenerContainerFactory.setAcknowledgeMode(AcknowledgeMode.MANUAL);

手动确认的消费者代码如下:

@SneakyThrows@RabbitListener(bindings =@QueueBinding(? ? ? ? exchange =@Exchange(value = RabbitMQConstant.CONFIRM_EXCHANGE, type = ExchangeTypes.TOPIC,? ? ? ? ? ? durable = RabbitMQConstant.FALSE_CONSTANT, autoDelete = RabbitMQConstant.true_CONSTANT),? ? ? ? value =@Queue(value = RabbitMQConstant.CONFIRM_QUEUE, durable = RabbitMQConstant.FALSE_CONSTANT,? ? ? ? ? ? autoDelete = RabbitMQConstant.true_CONSTANT),? ? ? ? key = RabbitMQConstant.CONFIRM_KEY),? ? ? ? containerFactory ="containerWithConfirm")publicvoidprocess(ExampleEvent msg, Channel channel, @Header(name ="amqp_deliveryTag")longdeliveryTag,? ? ? ? ? ? ? ? ? ? ? ? @Header("amqp_redelivered")booleanredelivered, @Headers Map head){try{? ? ? ? ? ? log.info("ConsumerWithConfirm receive message:{},header:{}", msg, head);? ? ? ? ? ? channel.basicAck(deliveryTag,false);? ? ? ? }catch(Exception e) {? ? ? ? ? ? log.error("consume confirm error!", e);//这一步千万不要忘记,不会会导致消息未确认,消息到达连接的qos之后便不能再接收新消息//一般重试肯定的有次数,这里简单的根据是否已经重发过来来决定重发。第二个参数表示是否重新分发channel.basicReject(deliveryTag, !redelivered);//这个方法我知道的是比上面多一个批量确认的参数// channel.basicNack(deliveryTag, false,!redelivered);}? ? }

关于spring的AcknowledgeMode需要说明,他一共有三种模式:NONE,MANUAL,AUTO,默认是AUTO模式。这比RabbitMq原生多了一种。这一点很容易混淆,这里的NONE对应其实就是RabbitMq的自动确认,MANUAL是手动。而AUTO其实也是手动模式,只不过是Spring的一层封装,他根据你方法执行的结果自动帮你发送ack和nack。如果方法未抛出异常,则发送ack。如果方法抛出异常,并且不是AmqpRejectAndDontRequeueException则发送nack,并且重新入队列。如果抛出异常时AmqpRejectAndDontRequeueException则发送nack不会重新入队列。我有一个例子专门测试NONE,见CunsumerWithNoneTest。

还有一点需要注意的是消费者有一个参数prefetch,它表示的是一个Channel(也就是SimpleMessageListenerContainer的一个线程)预取的消息数量,这个参数只会在手动确认的消费者才生效??梢钥突Ф死谜飧霾问刺岣咝阅芎妥隽髁靠刂啤H绻鹥refetch设置的是10,当这个Channel上unacked的消息数量到达10条时,RabbitMq便不会在向你发送消息,客户端如果处理的慢,便可以延迟确认在方法消息的接收。至于提高性能就非常容易理解,因为这个是批量获取消息,如果客户端处理的很快便不用一个一个去等着去新的消息。SpringAMQP2.0开始默认是250,这个参数应该已经足够了。注意之前的版本默认值是1所以有必要重新设置一下值。当然这个值也不能设置的太大,RabbitMq是通过round robin这个策略来做负载均衡的,如果设置的太大会导致消息不多时一下子积压到一台消费者,不能很好的均衡负载。另外如果消息数据量很大也应该适当减小这个值,这个值过大会导致客户端内存占用问题。如果你用到了事务的话也需要考虑这个值的影响,因为事务的用处不大,所以我也没做过多的深究。

关于发送者确认模式

考虑这样一个场景:你发送了一个消息给RabbitMq,RabbitMq接收了但是存入磁盘之前服务器就挂了,消息也就丢了。为了保证消息的投递有两种解决方案,最保险的就是事务(和DB的事务没有太大的可比性), 但是因为事务会极大的降低性能,会导致生产者和RabbitMq之间产生同步(等待确认),这也违背了我们使用RabbitMq的初衷。所以一般很少采用,这就引入第二种方案:发送者确认模式。

发送者确认模式是指发送方发送的消息都带有一个id,RabbitMq会将消息持久化到磁盘之后通知生产者消息已经成功投递,如果因为RabbitMq内部的错误会发送ack。注意这里的发送者和RabbitMq之间是异步的,所以相较于事务机制性能大大提高。其实很多操作都是不能保证绝对的百分之一百的成功,哪怕采用了事务也是如此,可靠性和性能很多时候需要做一些取舍,想很多互联网公司吹嘘的5个9,6个9也是一样的道理。如果不是重要的消息性能计数器,完全可以不采用发送者确认模式。

这里有一点我当时纠结了很久,我一直以为发送者确认模式的回调是客户端的ack触发的,这里是大大的误解!发送者确认模式和消费者没有一点关系,消费者确认也和发送者没有一点关系,两者都是在和RabbitMq打交道,发送者不会管消费者有没有收到,只要消息到了RabbitMq并且已经持久化便会通知生产者,这个ack是RabbitMq本身发出的,和消费者无关

发送者确认模式需要将Channel设置成Confirm模式,这样才会收到通知。Spring中需要将连接设置成Confirm模式:

connectionFactory.setPublisherConfirms(isConfirm);

然后在RabbitTemplate中设置确认的回调,correlationData是消息的id,如下(只是简单打印下):

// 设置RabbitTemplate每次发送消息都会回调这个方法rabbitTemplate.setConfirmCallback((correlationData, ack, cause)? ? ? ? ? ? -> log.info("confirm callback id:{},ack:{},cause:{}", correlationData, ack, cause));

发送时需要给出唯一的标识(CorrelationData):

rabbitTemplateWithConfirm.convertAndSend(RabbitMQConstant.DEFAULT_EXCHANGE, RabbitMQConstant.DEFAULT_KEY,newExampleEvent(i,"confirm message id:"+ i),newCorrelationData(Integer.toString(i)));

还有一个参数需要说下:mandatory。这个参数为true表示如果发送消息到了RabbitMq,没有对应该消息的队列。那么会将消息返回给生产者,此时仍然会发送ack确认消息。

设置RabbitTemplate的回调如下:

rabbitTemplate.setReturnCallback((message, replyCode, replyText, exchange, routingKey)? ? ? ? ? ? -> log.info("return callback message:{},code:{},text:{}", message, replyCode, replyText));

另外如果是RabbitMq内部的错误,不会调用该方法。所以如果消息特别重要,对于未确认的消息,生产者应该在内存用保存着,在确认时候根据返回的id删除该消息。如果是nack可以将该消息记录专门的日志或者转发到相应处理的逻辑进行后续补偿。RabbitTemplate也可以配置RetryTemplate,发送失败时直接进行重试,具体还是要结合业务。

最后关于发送者确认需要提的是spring,因为spring默认的Bean是单例的,所以针对不同的确认方案(其实有不同的确认方案是比较合理的,很多消息不需要确认,有些需要确认)需要配置不同的bean.

消费消息、死信队列和RetryTemplate

上面也提到了如果消费者抛出异常时默认的处理逻辑。另外我们还可以给消费者配置RetryTemplate,如果是采用SpringBoot的话,可以在application.yml配置中配置如下:

spring:? rabbitmq:? ? listener:? ? ? retry:#? ? 重试次数? ? ? ? ? max-attempts:3#? 开启重试机制? ? ? ? ? enabled:true

如上,如果消费者失败的话会进行重试,默认是3次。注意这里的重试机制RabbitMq是为感知的!到达3次之后会抛出异常调用MessageRecoverer。默认的实现为RejectAndDontRequeueRecoverer,也就是打印异常,发送nack,不会重新入队列。

我想既然配置了重试机制消息肯定是很重要的,消息肯定不能丢,仅仅是日志可能会因为日志滚动丢失而且信息不明显,所以我们要讲消息保存下来??梢杂腥缦抡庑┓桨福?/p>

使用RepublishMessageRecoverer这个MessageRecoverer会发送发送消息到指定队列

给队列绑定死信队列,因为默认的RepublishMessageRecoverer会发送nack并且requeue为false。这样抛出一场是这种方式和上面的结果一样都是转发到了另外一个队列。详见DeadLetterConsumer

注册自己实现的MessageRecoverer

给MessageListenerContainer设置RecoveryCallback

对于方法手动捕获异常,进行处理

我比较推荐前两种。这里说下死信队列,死信队列其实就是普通的队列,只不过一个队列声明的时候指定的属性,会将死信转发到该交换器中。声明死信队列方法如下:

@RabbitListener(? ? ? ? bindings =@QueueBinding(? ? ? ? ? ? exchange =@Exchange(value = RabbitMQConstant.DEFAULT_EXCHANGE, type = ExchangeTypes.TOPIC,? ? ? ? ? ? ? ? durable = RabbitMQConstant.FALSE_CONSTANT, autoDelete = RabbitMQConstant.true_CONSTANT),? ? ? ? ? ? value =@Queue(value = RabbitMQConstant.DEFAULT_QUEUE, durable = RabbitMQConstant.FALSE_CONSTANT,? ? ? ? ? ? ? ? autoDelete = RabbitMQConstant.true_CONSTANT, arguments = {@Argument(name = RabbitMQConstant.DEAD_LETTER_EXCHANGE, value = RabbitMQConstant.DEAD_EXCHANGE),@Argument(name = RabbitMQConstant.DEAD_LETTER_KEY, value = RabbitMQConstant.DEAD_KEY)? ? ? ? ? ? }),? ? ? ? ? ? key = RabbitMQConstant.DEFAULT_KEY? ? ? ? ))

其实也就只是在声明的时候多加了两个参数x-dead-letter-exchange和x-dead-letter-routing-key。这里一开始踩了一个坑,因为@QueueBinding注解中也有arguments属性,我一开始将参数声明到@QueueBinding中,导致一直没绑定成功。如果绑定成功可以在控制台看到queue的Featrues有DLX(死信队列交换器)和DLK(死信队列绑定)。如下:

关于消息进入死信的规则

消息被拒绝(basic.reject/basic.nack)并且requeue=false

消息TTL过期

队列达到最大长度

我们用到的就是第一种。

RPC模式的消息(不常用)

本来生产者和消费者是没有耦合的,但是可以通过一些属性产生耦合。在早期版本中,如果一个生产者想要收到消费者的回复,实现方案是生产者在消息头中加入reply-to属性也就是队列(一般是私有,排他,用完即销毁)的名字,然后在这个队列上进行监听,消费者将回复发送到这个队列中。RabbitMq3.3之后有了改进,就是不用没有都去创建一个临时队列,这样很耗费性能,可以采用drect-to模式,省去了每次创建队列的性能损耗,但是还是要创建一次队列。现在Spring默认的就是这个模式。RabbitTemplate中有一系列的sendAndReceiveXX方法。默认等待5秒,超时返回null。用

法和不带返回的差不多。

消费者的方法通过返回值直接返回消息(下面的方法是有返回值的):

publicStringreceive(@Headers Map<String, Object> headers, @Payload ExampleEvent msg){? ? ? ? log.info("reply to consumer param:{headers = ["+ headers +"], msg = ["+ msg +"]} info:");returnREPLY;? }

这里的提一下最后一个注解@SendTo,用在消费方法上,指明返回值的目的地,默认不用的话就是返回给发送者,可以通过这个注解改变这种行为。如下代码:

@RabbitListener(? ? ? ? bindings =@QueueBinding(? ? ? ? ? ? exchange =@Exchange(value = RabbitMQConstant.REPLY_EXCHANGE, type = ExchangeTypes.TOPIC,? ? ? ? ? ? ? ? durable = RabbitMQConstant.FALSE_CONSTANT, autoDelete = RabbitMQConstant.true_CONSTANT),? ? ? ? ? ? value =@Queue(value = RabbitMQConstant.REPLY_QUEUE, durable = RabbitMQConstant.FALSE_CONSTANT,? ? ? ? ? ? ? ? autoDelete = RabbitMQConstant.true_CONSTANT),? ? ? ? ? ? key = RabbitMQConstant.REPLY_KEY? ? ? ? )? ? )@SendTo("queue.reply.s")publicExampleEventlog(ExampleEvent event){? ? ? ? log.info("log receive message:O{}", event);returnnewExampleEvent(1,"log result");? ? }

上面的代码就是会将消息直接发送到默认交换器,并且以queue.reply.s作为路由键。@SendTo的格式为exchange/routingKey用法如下:

foo/bar: 指定的交换器和key

foo/: 指定的交换器,key为空

bar或者/bar: 到空交换器

/或者空:空的交换器和空的key

这里还需要提一下,因为默认所有的队列都会绑定到空交换器,并且以队列名字作为Routekey, 所以SendTo里面可以直接填写队列名字机会发送到相应的队列.如日志队列。因为RPC模式不常用,专业的东西做专业的事,就像我们一般不用Redis来做消息队列一样(虽然他也可以实现),一般公司都有特定的技术栈,肯定有更合适的RPC通信框架。当然如果要跨语言的集成这个方案也是一种不错的方案,可以继续考虑采用异步发送AsyncRabbitTemplate来降低延迟等优化方案!

关于消费模型

RabbitMQ底层的消费模型有两种Push和Pull。我在网上查阅资料的时候发现有很多教程采用了pull这种模式。RabbitMq实战和

RabbitMQ之Consumer消费模式(Push & Pull)都指出这种模式性能低,会影响消息的吞吐量,增加不必要的IO,所以除非有特殊的业务需求,不要采用这种方案。Spring的封装就是采用了push的方案。

关于RabbitMq客户端的线程模型

这里讲的是消费者的,生产者没什么好讲的。先看消息流转图:

图中椭圆表示线程,矩形是队列。消息到达AMQP的连接线程,然后分发到client线程池,随后分发到监听器。注意除了监听器的线程,其他都是在com.rabbitmq.client.impl.AMQConnection中创建的线程,我们对线程池做一些修改。连接线程名字不能修改就是AMQP Connection打头。心跳线程可以设置setConnectionThreadFactory来设置名字。如下:

connectionFactory.setConnectionThreadFactory(newThreadFactory() {publicfinalAtomicInteger id =newAtomicInteger();@OverridepublicThreadnewThread(Runnable r){returnnewThread(r, MessageFormat.format("amqp-heart-{0}", id.getAndIncrement()));? ? ? ? ? ? }? ? ? ? });

client线程池见:com.rabbitmq.client.impl.ConsumerWorkService构造方法。Executors.newFixedThreadPool(DEFAULT_NUM_THREADS, threadFactory)。

finalExecutorService executorService = Executors.newFixedThreadPool(5,newThreadFactory() {publicfinalAtomicInteger id =newAtomicInteger();@OverridepublicThreadnewThread(Runnable r){returnnewThread(r, MessageFormat.format("amqp-client-{0}", id.getAndIncrement()));? ? ? ? ? ? }? ? ? ? });

listener的线程设置如下:

simpleRabbitListenerContainerFactory.setTaskExecutor(newSimpleAsyncTaskExecutor"amqp-consumer-"));

注意:SimpleAsyncTaskExecutor每次执行一个任务都会新建一个线程,对于生命周期很短的任务不要使用这个线程池(如client线程池的任务), 这里的消费者线程生命周期直到SimpleMessageListenerContainer停止所以没有适合这个场景

修改过之后的线程如下:

消息投递过程如下:

在AMQConnection中开启连接线程,该线程用于处理和RabbitMq的通信:

publicvoidstartMainLoop(){? ? ? ? MainLoop loop =newMainLoop();finalString name ="AMQP Connection "+ getHostAddress() +":"+ getPort();? ? ? ? mainLoopThread = Environment.newThread(threadFactory, loop, name);? ? ? ? mainLoopThread.start();? ? }

AMQConnection.heartbeatExecutor是心跳线程。

AMQConnection.consumerWorkServiceExecutor则是用来处理事件的线程池,AMQConnection线程收到消息投递到这里。

分发逻辑详见com.rabbitmq.client.impl.ChannelN#processAsync->com.rabbitmq.client.impl.ConsumerDispatcher#handleDelivery->投递到线程池.

线程池中继续将消息投递到org.springframework.amqp.rabbit.listener.BlockingQueueConsumer#queue中

consumer线程进行最终消息

上面的是默认的消费者监听器。SpringAMQP 2.0引入了一个新的监听器实现DirectMessageListenerContainer。这个实现最大的变化在于消费者的处理逻辑不是在自己的线程池中执行而是直接在client线程池中处理,这样最明显的是省去了线程的上下文切换的开销,而且设计上也变得更为直观。所以如果采用这个监听器需要覆盖默认的线程池加大Connection的线程池。采用这个监听器只需要设置@RabbitListener的containerFactory属性。声明方法如下:

@BeanDirectRabbitListenerContainerFactorydirectRabbitListenerContainerFactory(ConnectionFactory connectionFactory){finalDirectRabbitListenerContainerFactory directRabbitListenerContainerFactory =newDirectRabbitListenerContainerFactory();? ? ? ? directRabbitListenerContainerFactory.setConsumersPerQueue(Runtime.getRuntime().availableProcessors());? ? ? ? directRabbitListenerContainerFactory.setConnectionFactory(connectionFactory);? ? ? ? directRabbitListenerContainerFactory.setMessageConverter(newJackson2JsonMessageConverter());? ? ? ? directRabbitListenerContainerFactory.setConsumersPerQueue(10);returndirectRabbitListenerContainerFactory;? ? }

这时的消息流转图如下:

还有一些关于监听器的例子和Springboot配置我放在了源码里,这里不再讲述。

这篇文章就总结到这里,希望能够对你有所帮助! ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?欢迎工作一到五年的Java工程师朋友们加入Java群:?891219277

群内提供免费的Java架构学习资料(里面有高可用、高并发、高性能及分布式、Jvm性能调优、Spring源码,MyBatis,Netty,Redis,Kafka,Mysql,Zookeeper,Tomcat,Docker,Dubbo,Nginx等多个知识点的架构资料)合理利用自己每一分每一秒的时间来学习提升自己,不要再用"没有时间“来掩饰自己思想上的懒惰!趁年轻,使劲拼,给未来的自己一个交代!

?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,172评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,346评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事?!?“怎么了?”我有些...
    开封第一讲书人阅读 159,788评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,299评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,409评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,467评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,476评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,262评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,699评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,994评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,167评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,499评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,149评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,387评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,028评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,055评论 2 352

推荐阅读更多精彩内容