一. 持久化
1.1 为什么进行持久化?
防止RabbitMQ客户端出现问题导致数据丢失
1.2 如何避免消息丢失?
- 消费者的手动ACK机制。可以防止业务处理失败。
- 但是,如果在消费者消费之前,MQ就宕机了,消息就没了。
1.3 如何持久化
要将消息持久化,前提是:队列、Exchange都持久化
1.3.1 交换机持久化
1.3.2 队列持久化
1.3.3 消息持久化
1.3.4 测试
分别测试持久化和非持久化:
1、Send给Recv发送50条消息
2、Recv收到一条消息sleep1秒钟,收到前几条消息后立即关闭
3、重启RabbitMQ观察消息是否丢失
二. Spring AMQP
2.1 Spring AMQP简介
Spring有很多不同的项目,其中就有对AMQP的支持:
Spring AMQP的页面:Spring AMQP
注意这里一段描述:
Spring-amqp是对AMQP协议的抽象实现,而spring-rabbit 是对协议的具体实现,也是目前的唯一实现。底层使用的就是RabbitMQ。
2.2 依赖和配置
添加AMQP的启动器:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-amqp</artifactId>
</dependency>
在application.properties
中添加RabbitMQ地址:
#主机
spring.rabbitmq.host=192.168.233.132
#端口
spring.rabbitmq.port=5672
#用户名
spring.rabbitmq.username=admin
#密码
spring.rabbitmq.password=1111
#虚拟分组
spring.rabbitmq.virtual-host=/
2.3 监听者
在SpringAmqp中,对消息的消费者进行了封装和抽象,一个普通的JavaBean中的普通方法,只要通过简单的注解,就可以成为一个消费者。
@Component
public class Listener {
/**
* 监听者接收消息三要素:
* 1、queue
* 2、exchange
* 3、routing key
*/
@RabbitListener(bindings = @QueueBinding(
value = @Queue(value="springboot_queue",durable = "true"),
exchange = @Exchange(value="springboot_exchage",type= ExchangeTypes.TOPIC),
key= {"*.*"}
))
public void listen(String msg){
System.out.println("接收到消息:" + msg);
}
}
-
@Componet
:类上的注解,注册到Spring容器 -
@RabbitListener
:方法上的注解,声明这个方法是一个消费者方法,需要指定下面的属性:-
bindings
:指定绑定关系,可以有多个。值是@QueueBinding
的数组。@QueueBinding
包含下面属性:-
value
:这个消费者关联的队列。值是@Queue
,代表一个队列 -
exchange
:队列所绑定的交换机,值是@Exchange
类型 -
key
:队列和交换机绑定的RoutingKey
-
-
类似listen这样的方法在一个类中可以写多个,就代表多个消费者
2.4 AmqpTemplate
Spring最擅长的事情就是封装,把他人的框架进行封装和整合。
Spring为AMQP提供了统一的消息处理模板:AmqpTemplate,非常方便的发送消息,其发送方法:
红框圈起来的是比较常用的3个方法,分别是:
-
指定交换机、RoutingKey和消息体
-
指定消息
-
指定RoutingKey和消息,会向默认的交换机发送消息
2.5 发送者
@RunWith(SpringRunner.class)
@SpringBootTest(classes = App.class)
public class MqDemo {
@Autowired
private AmqpTemplate amqpTemplate;
@Test
public void testSend() throws InterruptedException {
String msg = "hello, Spring boot amqp";
this.amqpTemplate.convertAndSend("spring.test.exchange","a.b", msg);
// 等待10秒后再结束
Thread.sleep(10000);
}
}
运行后查看日志:
2.6 手动ack
2.6.1 application.properties
#设置三种订阅模式手动ack
spring.rabbitmq.listener.direct.acknowledge-mode=manual
#设置work消息类型手动ack
spring.rabbitmq.listener.simple.acknowledge-mode=manual
2.6.2 监听者
监听方法添加Channel channel, Message message两个参数
@Component
public class Recv {
/**
* 监听者接收消息三要素:
* 1、queue
* 2、exchange
* 3、routing key
*/
@RabbitListener(bindings = @QueueBinding(
value = @Queue(value="springboot_queue",durable = "true"),
exchange = @Exchange(value="springboot_exchage",type= ExchangeTypes.TOPIC),
key= {"*.*"}
))
public void listen(String msg, Channel channel, Message message) throws IOException {
System.out.println("接收到消息:" + msg);
//int a = 6/0;
channel.basicAck(message.getMessageProperties().getDeliveryTag(),false);
}
}