RabbitMQ 是一个由 Erlang 语言开发的 AMQP 的开源实现。
AMQP :Advanced Message Queue,高级消息队列协议。它是应用层协议的一个开放标准,为面向消息的中间件设计,基于此协议的客户端与消息中间件可传递消息,并不受产品、开发语言等条件的限制。
可靠性(Reliability)
RabbitMQ 使用一些机制来保证可靠性,如持久化、传输确认、发布确认。
灵活的路由(Flexible Routing)
在消息进入队列之前,通过 Exchange 来路由消息的。对于典型的路由功能,RabbitMQ 已经提供了一些内置的 Exchange 来实现。针对更复杂的路由功能,可以将多个 Exchange 绑定在一起,也通过插件机制实现自己的 Exchange 。
消息集群(Clustering)
多个 RabbitMQ 服务器可以组成一个集群,形成一个逻辑 Broker 。
高可用(Highly Available Queues)
队列可以在集群中的机器上进行镜像,使得在部分节点出问题的情况下队列仍然可用。
多种协议(Multi-protocol)
RabbitMQ 支持多种消息队列协议,比如 STOMP、MQTT 等多种消息中间件协议。
多语言客户端(Many Clients)
RabbitMQ 几乎支持所有常用语言,比如Java、Python 、Ruby 、PHP 、C# 、JavaScript 等。
管理界面(Management UI)
RabbitMQ 提供了一个易用的用户界面,使得用户可以监控和管理消息 Broker 的许多方面。
跟踪机制(Tracing)
如果消息异常,RabbitMQ 提供了消息跟踪机制,使用者可以找出发生了什么。
插件机制(Plugin System)
RabbitMQ 提供了许多插件,来从多方面进行扩展,也可以编写自己的插件。
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<optional>true</optional>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-amqp</artifactId>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
</dependency>
只要引入了amqp场景,就会有相关的自动配置AutoConfiguration类
用idea搜索RabbitAutoConfiguration源码如下:
可以看到这个类会给容器中自动配置了RabbitTemplate ,AmqpAdmin,CachingConnectionFactory等等.
连接Rabbit,RabbitAutoConfiguration可以找到连接工厂可以看到从RabbitProperties类找到所有连接信息进入源码如下图:
可以看到很多都有默认值
yml配置
spring:
rabbitmq:
host: //填自己主机地址
port: 5672
virtual-host: /
import org.springframework.amqp.support.converter.Jackson2JsonMessageConverter;
import org.springframework.amqp.support.converter.MessageConverter;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
public class RabbitConfig {
@Bean
public MessageConverter messageConverter() {
return new Jackson2JsonMessageConverter();
}
}
@Service
public class Consumer {
/**
* queues 需要监听的队列
*
* message: 原生消息详细信息。头+体
* T<发送的消息类型>> Order order
* channel:当前传输数据的通道
*/
@RabbitListener(queues = {"java-Queue"})
public void recieveMessage(Object message,
Order order,
Channel channel) {
System.out.println(message);
System.out.println(order);
}
}
spring:
rabbitmq:
host:
port: 5672
virtual-host: /
#开启发送端确认
publisher-confirm-type: correlated
@PostConstruct
public void initRabbitTemplate() {
//设置一个确认回调
rabbitTemplate.setConfirmCallback(new RabbitTemplate.ConfirmCallback() {
/**
*
* @param correlationData 当前消息唯一关联数据
* @param b 是否成功收到
* @param cause 失败的原因
*/
@Override
public void confirm(CorrelationData correlationData, boolean b, String cause) {
log.info("correlationData:{},ack:{},cause:{}", correlationData, b, cause);
}
});
}
spring:
rabbitmq:
host:
port: 5672
virtual-host: /
#开启发送端确认
publisher-confirm-type: correlated
#开启发送消息抵达队列的确认
publisher-returns: true
#只要抵达队列,以异步发送优先回调returnConfirm
template:
mandatory: true
@PostConstruct
public void initRabbitTemplate() {
//设置一个确认回调
rabbitTemplate.setConfirmCallback(new RabbitTemplate.ConfirmCallback() {
/**
* @param correlationData 当前消息唯一关联数据
* @param b 是否成功收到
* @param cause 失败的原因
*/
@Override
public void confirm(CorrelationData correlationData, boolean b, String cause) {
log.info("correlationData:{},ack:{},cause:{}", correlationData, b, cause);
}
});
//设置消息抵达队列的确认回调
rabbitTemplate.setReturnsCallback(new RabbitTemplate.ReturnsCallback() {
//只要消息没有投递到指定队列,就回触发这个失败回调
/**
*
* @param returnedMessage 该类有以下变量
* Message message; 投递失败的消息详细信息
* replyCode; 回复的状态吗
* replyText; 回复的文本内容
* exchange; 当时这个消息发给那个交换机
* routingKey; 当时这个消息用那个路由键
*/
@Override
public void returnedMessage(ReturnedMessage returnedMessage) {
log.info("message:{},replyCode:{},replyText:{},exchange:{},routingKey:{}",
returnedMessage.getMessage(),
returnedMessage.getReplyCode(),
returnedMessage.getReplyText(),
returnedMessage.getExchange(),
returnedMessage.getRoutingKey());
}
});
}
message:(Body:'{"id":1,"userId":1}' MessageProperties [headers={__TypeId__=com.mq.rabbit.entity.Order}, contentType=application/json, contentEncoding=UTF-8, contentLength=0, receivedDeliveryMode=PERSISTENT, priority=0, deliveryTag=0]),replyCode:312,replyText:NO_ROUTE,exchange:java-Exchange,routingKey:java-error
开启消息手动确认(acknowledge:manual)
spring:
rabbitmq:
host:
port: 5672
virtual-host: /
#开启发送端确认
publisher-confirm-type: correlated
#开启发送消息抵达队列的确认
publisher-returns: true
#只要抵达队列,以异步发送优先回调returnConfirm
template:
mandatory: true
#默认客户端接收到消息自动ack, acknowledge修改为manual 开启手动确认
listener:
simple:
acknowledge-mode: manual
/**
* queues 需要监听的队列
*
* message: 原生消息详细信息。头+体
* T<发送的消息类型>> Order order
* channel:当前传输数据的通道
*/
@RabbitHandler
public void recieveMessage(Message message, Order order,
Channel channel) throws IOException {
//deliveryTag channel内按顺序自增
long deliveryTag = message.getMessageProperties().getDeliveryTag();
log.info("接收队列订单消息{}", order);
if (ThreadLocalRandom.current().nextInt(100) % 2 == 0) {
/**签收消息
* long deliveryTag, boolean multiple
* multiple 是否批量签收模式 false代表我只签收我当前这条消息
* 签收需要给broker回复ack,所以有可能网络中断 需要捕获异常或者抛出异常
*/
channel.basicAck(deliveryTag, false);
log.info("消息签收成功");
} else {
/**
* long deliveryTag, boolean multiple, boolean requeue
* multiple 是否批量拒绝签收
* requeue true 发回服务器,重新入队 false 丢弃
*/
channel.basicNack(deliveryTag, false, false);
log.info("消息签收失败丢弃消息");
}
}
完
感谢您的阅读
如果你发现了错误的地方,可以在留言区提出来,我对其加以修改