数据丢失的原因
1.存储在队列中,如果队列没有对消息持久化,RabbitMQ服务器宕机重启会丢失数据。
2.生产者发送消息到RabbitMQ服务器过程中,RabbitMQ服务器如果宕机停止服务,消息会丢失。
3.消费者从RabbitMQ服务器获取队列中存储的数据消费,但是消费者程序出错或者宕机而没有正确消费,导致数据丢失。
解决
1.持久化是默认开启的
2.开启消息确认机制 confirm机制
一条消息从生产者发送到RabbitMQ,首先会发送到Exchange,对应回调函数confirm()。第二步从Exchange路由分配到Queue中,对应回调函数则是returnedMessage()。
@Bean(name = "baseConnectionFactory")
public ConnectionFactory getCreditFactory() {
CachingConnectionFactory connectionFactory = new CachingConnectionFactory();
connectionFactory.setAddresses(host + ":" + port);
connectionFactory.setUsername(username);
connectionFactory.setPassword(password);
connectionFactory.setVirtualHost("/");
// 发布确认,收到客户端响应才会被删除消息
connectionFactory.setPublisherConfirms(true);
return connectionFactory;
}
3.开启ack确认机制
把自动签收改成手动签收,正常消费则返回确认签收,如果出现异常,则返回拒绝签收重回队列。
@Bean(name = "baseContainerFactory")
public SimpleRabbitListenerContainerFactory firstFactory(SimpleRabbitListenerContainerFactoryConfigurer
configurer, @Qualifier("baseConnectionFactory") ConnectionFactory connectionFactory) {
SimpleRabbitListenerContainerFactory factory = new SimpleRabbitListenerContainerFactory();
// 设置手动确认消费
factory.setAcknowledgeMode(AcknowledgeMode.MANUAL);
// 设置消费者个数
factory.setConcurrentConsumers(6);
configurer.configure(factory, connectionFactory);
return factory;
}
消费者监听器时:
protected void messageAck(Message message, Channel channel, boolean lockInd, String mqInfo) {
try {
if (lockInd) {
//basicAck(该消息的index,是否批量)
channel.basicAck(message.getMessageProperties().getDeliveryTag(), false);
log.info("mq消息:{}确认成功", mqInfo);
} else {
//basicReject(该消息的index,被拒绝是否重新入队列)
channel.basicReject(message.getMessageProperties().getDeliveryTag(), true);
log.info("重回队列", mqInfo);
}
} catch (IOException e) {
log.info("消息确认异常,message:{}", message.getBody(), e);
}
}