目录
第九章-RabbitMQ整合SpringBoot
1. 前言
2. 整合方式
2.1 pom文件
2.2 配置文件
2.3 生产者
2.4 消费者
3. 整合案例
3.1 helloword
3.2 work
3.3 pub/sub
3.4 routing
4. 总结
前面学了很多都是基于java 的AMQP client 的内容,但到实际开发中运用呢,相信没有同学直接使用amqp client 吧。
所以,最终还是需要回归到java 的王者Spring,无论spring也好,还是SpringBoot 都是基于amqp client再做了一层封装而已,为的也是让大家用的爽,专注于业务。
由于大家基本SpringBoot用的最多,我们也就只涉及关于SpringBoot 的整合。用的时候肯定是用Springboot封装的方式,但如果大家想要系统的学的话,建议还是要基于Amqp Client去学,循序渐进,才能明白,SpringBoot 封装的那些注解,是啥东东。
org.springframework.boot
spring-boot-starter-amqp
spring.rabbitmq.host = 127.0.0.1
spring.rabbitmq.virtual-host = my-test-virtual
spring.rabbitmq.port = 5672
spring.rabbitmq.username = test
spring.rabbitmq.password = test
#spring.rabbitmq.listener.simple.acknowledge-mode = auto
#spring.rabbitmq.listener.simple.prefetch = 1
根据发送到不同的Exchange,代码不同
// 发送到默认Exchange
rabbitTemplate.convertAndSend("SolarWaterHeater", msg);
//发送到指定的Exchange
rabbitTemplate.convertAndSend(exchangeName, routeKey, msg);
根据Exchange的不同,队列的是否存在,消费者使用的注解方式不同
1、队列已经存在且相关的Exchange和绑定关系都OK的话,直接使用:
@RabbitHandler
@RabbitListener(queues = "SolarWaterHeater")
public void receiveBAmtInMessage(Message message, Channel channel) throws InterruptedException {
Thread.sleep(1000);
log.info("小明洗澡用水: " + new String(message.getBody()));
}
2. 如果使用默认的Exchange,且队列不存在的话,需要在注解上声明Queue的情况
@RabbitHandler
@RabbitListener(queuesToDeclare = @Queue("SolarWaterHeater"))
public void receiveBAmtInMessage(Message message, Channel channel) throws InterruptedException {
Thread.sleep(1000);
log.info("小明洗澡用水: " + new String(message.getBody()));
}
3. 使用配置类,声明Queue
@Configuration
public class QueueConfig {
/**
* 定义一个队列
*
* @return
*/
@Bean("SolarWaterHeater")
public Queue solarWaterHeater() {
/*
第二种方式:
通过new Queue对象来创建队列
参数1: 队列名称
参数2: 是否持久化(默认:true)
参数3: 是否独占(默认:false)
参数4: 是否自动删除(默认:false)
参数5: 队列的其他参数
*/
return new Queue("SolarWaterHeater", true, false, false, null);
}
/* @Bean("SolarWaterHeater")
public Queue solarWaterHeater() {
*//*
第一种方式:
durable():代表需要持久化
exclusive(): 代表该队列独占(只允许有一个consumer监听)
autoDelete(): 代表需要自动删除(没有consumer自动删除)
withArgument(): 队列的其他参数
*//*
return QueueBuilder.durable("boot_work_queue").exclusive().autoDelete().withArgument("key", "val").build();
}*/
}
4. 需要声明Exchange并绑定Queue的情况
@RabbitHandler
@RabbitListener(bindings = @QueueBinding(
value = @Queue("SolarWaterHeater-RedWine"),
key = "REDWINE",
exchange = @Exchange(value = "routing-exchange", type = ExchangeTypes.DIRECT, durable = "false")))
public void receiveXMInMessage(Message message, Channel channel) throws InterruptedException {
Thread.sleep(1000);
log.info("小明洗澡用: " + new String(message.getBody()));
}
5. 使用配置的方式:
@Configuration
public class RoutingConfig {
/**
* 准备两个队列SolarWaterHeater-RedWine、SolarWaterHeater-Milk
*
* @return
*/
@Bean("SolarWaterHeater-RedWine")
public Queue bootDirectQueue1() {
return QueueBuilder.durable("SolarWaterHeater-RedWine").build();
}
@Bean("SolarWaterHeater-Milk")
public Queue bootDirectQueue2() {
return QueueBuilder.durable("SolarWaterHeater-Milk").build();
}
// direct类型交换机
@Bean("routing-exchange")
public Exchange bootDirectExchange() {
/*
第一种方式: 通过ExchangeBuilder构建交换机
durable: 是否持久化
autoDelete: 是否自动删除
withArgument: 交换机其他参数
*/
// return ExchangeBuilder.directExchange("boot_direct_exchange").durable(true).autoDelete().withArgument("key","val").build();
/*
第二种方式:
参数1: 是否持久化(默认false)
参数2: 是否自动删除(默认false)
参数3: 其他参数
*/
return new DirectExchange("routing-exchange", true, false, null);
}
/**
* 交换机与队列进行绑定
*/
@Bean
public Binding bindDirect1() {
/*
第一种方式:
bind(Queue): 需要绑定的queue
to(Exchange): 需要绑定到哪个交换机
with(String): routing key
noargs(): 进行构建
*/
// return BindingBuilder.bind(bootDirectQueue1).to(bootDirectExchange).with("article").noargs();
/*
第一种方式:
参数1: 绑定的队列
参数2: 绑定的类型 Binding.DestinationType.QUEUE: 绑定的类型为queue(交换机不仅可以绑定queue还可以绑定exchange)
参数3: 哪个交换机需要绑定
参数4: routing key
参数5: 其他参数
*/
return new Binding("SolarWaterHeater-RedWine", Binding.DestinationType.QUEUE, "routing-exchange", "REDWINE", null);
}
/**
* 交换机与队列进行绑定
*
* @return
*/
@Bean
public Binding bindDirect2() {
Queue bootDirectQueue2 = bootDirectQueue2();
Exchange bootDirectExchange = bootDirectExchange();
return BindingBuilder.bind(bootDirectQueue2).to(bootDirectExchange).with("MILK").noargs();
}
}
生产者:
@RestController
@RequestMapping("/mq")
@Slf4j
public class WaterProducerController {
@Resource
private RabbitTemplate rabbitTemplate;
@PostMapping("/send")
public void send() {
try {
for (int i = 1; i <= 50; i++) {
String msg = i + "升";
rabbitTemplate.convertAndSend("SolarWaterHeater", msg);
Thread.sleep(1000);
log.info("水龙头放水成功!" + i + "升");
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
消费者:
@Slf4j
@Component
public class XMShowerConsumer {
@RabbitHandler
@RabbitListener(queuesToDeclare = @Queue("SolarWaterHeater"))
public void receiveBAmtInMessage(Message message, Channel channel) throws InterruptedException {
Thread.sleep(1000);
log.info("小明洗澡用水: " + new String(message.getBody()));
}
}
平均模式:
生产者:
@RestController
@RequestMapping("/mqAverage")
@Slf4j
public class WaterProducerAverageController {
@Resource
private RabbitTemplate rabbitTemplate;
@PostMapping("/send")
public void send() {
try {
for (int i = 1; i <= 50; i++) {
String msg = i + "升";
rabbitTemplate.convertAndSend("SolarWaterHeater-Average", msg);
Thread.sleep(100);
log.info("水龙头放水成功!" + i + "升");
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
消费者:
/**
* @author rabbit
* @version 1.0.0
* @Description 平均模式
* 需要配置 :
* * spring.rabbitmq.listener.simple.acknowledge-mode = auto
* @createTime 2022/08/27 20:22:00
*/
@Slf4j
@Component
public class AverageShowerConsumer {
@RabbitHandler
@RabbitListener(queuesToDeclare = @Queue("SolarWaterHeater-Average"))
public void receiveXMInMessage(Message message, Channel channel) throws InterruptedException {
Thread.sleep(300);
log.info("小明洗澡用水: " + new String(message.getBody()));
}
@RabbitHandler
@RabbitListener(queuesToDeclare = @Queue("SolarWaterHeater-Average"))
public void receiveXLInMessage(Message message, Channel channel) throws InterruptedException {
Thread.sleep(1000);
log.info("小丽洗澡用水: " + new String(message.getBody()));
}
}
公平模式:
生产者:
@RestController
@RequestMapping("/mqFair")
@Slf4j
public class WaterProducerFairController {
@Resource
private RabbitTemplate rabbitTemplate;
@PostMapping("/send")
public void send() {
try {
for (int i = 1; i <= 50; i++) {
String msg = i + "升";
rabbitTemplate.convertAndSend("SolarWaterHeater-Fair", msg);
Thread.sleep(100);
log.info("水龙头放水成功!" + i + "升");
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
消费者:
/**
* @author rabbit
* @version 1.0.0
* @Description 公平模式
* 需要配置 :
* spring.rabbitmq.listener.simple.acknowledge-mode = manual
* spring.rabbitmq.listener.simple.prefetch = 1
* 并 消费后进行手动确认
* @createTime 2022/08/27 20:22:00
*/
@Slf4j
@Component
public class FairShowerConsumer {
@RabbitHandler
@RabbitListener(queuesToDeclare = @Queue("SolarWaterHeater-Fair"))
public void receiveXMInMessage(Message message, Channel channel) throws InterruptedException {
Thread.sleep(1000);
log.info("小明洗澡用水: " + new String(message.getBody()));
try {
final long deliveryTag = message.getMessageProperties().getDeliveryTag();
channel.basicAck(deliveryTag, false);
} catch (IOException e) {
log.error(" basicAck mq处理失败", e);
}
}
@RabbitHandler
@RabbitListener(queuesToDeclare = @Queue("SolarWaterHeater-Fair"))
public void receiveXLInMessage(Message message, Channel channel) throws InterruptedException {
Thread.sleep(500);
log.info("小丽洗澡用水: " + new String(message.getBody()));
try {
final long deliveryTag = message.getMessageProperties().getDeliveryTag();
channel.basicAck(deliveryTag, false);
} catch (IOException e) {
log.error(" basicAck mq处理失败", e);
}
}
}
生产者:
@RestController
@RequestMapping("/mqPublish")
@Slf4j
public class WaterPublishProducerController {
@Resource
private RabbitTemplate rabbitTemplate;
@PostMapping("/send")
public void send() {
try {
for (int i = 1; i <= 50; i++) {
String msg = i + "升";
//交换机名称
String exchangeName = "publish-exchange";
//路由key 由于我们实现的是fanout模式(广播模式),不需要路由key,所有的消费者都可以进行监听和消费
String routeKey = "";
//发送mq消息
rabbitTemplate.convertAndSend(exchangeName, routeKey, msg);
Thread.sleep(1000);
log.info("水龙头放水成功!" + i + "升");
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
消费者:
/**
* @author rabbit
* @version 1.0.0
* @Description 广播模式
* 需要配置 :
* * spring.rabbitmq.listener.simple.acknowledge-mode = auto
* @createTime 2022/08/27 20:22:00
*/
@Slf4j
@Component
public class PublishShowerConsumer {
@RabbitHandler
@RabbitListener(bindings = @QueueBinding(
value = @Queue("SolarWaterHeaterXM"),
exchange = @Exchange(value = "publish-exchange", type = ExchangeTypes.FANOUT, durable = "false")))
public void receiveXMInMessage(Message message, Channel channel) throws InterruptedException {
Thread.sleep(1000);
log.info("小明洗澡用水: " + new String(message.getBody()));
}
@RabbitHandler
@RabbitListener(bindings = @QueueBinding(
value = @Queue("SolarWaterHeaterXL"),
exchange = @Exchange(value = "publish-exchange", type = ExchangeTypes.FANOUT, durable = "false")))
public void receiveXLInMessage(Message message, Channel channel) throws InterruptedException {
Thread.sleep(1000);
log.info("小丽洗澡用水: " + new String(message.getBody()));
}
}
生产者:
@RestController
@RequestMapping("/mqRouting")
@Slf4j
public class WaterRoutingProducerController {
@Resource
private RabbitTemplate rabbitTemplate;
@PostMapping("/send")
public void send() {
try {
for (int i = 1; i <= 50; i++) {
String msg1 = "牛奶" + i + "升";
String msg2 = "红酒" + i + "升";
//交换机名称
String exchangeName = "routing-exchange";
//发送mq消息
rabbitTemplate.convertAndSend(exchangeName, "MILK", msg1);
rabbitTemplate.convertAndSend(exchangeName, "REDWINE", msg2);
Thread.sleep(1000);
log.info("水龙头放牛奶和红酒成功!" + i + "升");
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
消费者:
/**
* @author rabbit
* @version 1.0.0
* @Description 路由模式
* @createTime 2022/08/27 20:22:00
*/
@Slf4j
@Component
public class RoutingShowerConsumer {
@RabbitHandler
@RabbitListener(bindings = @QueueBinding(
value = @Queue("SolarWaterHeater-RedWine"),
key = "REDWINE",
exchange = @Exchange(value = "routing-exchange", type = ExchangeTypes.DIRECT, durable = "false")))
public void receiveXMInMessage(Message message, Channel channel) throws InterruptedException {
Thread.sleep(1000);
log.info("小明洗澡用: " + new String(message.getBody()));
}
@RabbitHandler
@RabbitListener(bindings = @QueueBinding(
value = @Queue("SolarWaterHeater-Milk"),
key = "MILK",
exchange = @Exchange(value = "routing-exchange", type = ExchangeTypes.DIRECT, durable = "false")))
public void receiveXLInMessage(Message message, Channel channel) throws InterruptedException {
Thread.sleep(1000);
log.info("小丽洗澡用: " + new String(message.getBody()));
}
}
无论是使用java amqp client 还是 spring boot 无外乎还是做那几件事:
1.创建连接
2. 声明Exchange
3. 声明queue
4. 创建绑定关系
5. 发送
6. 消费
只不过是有的需要自己手动做,有的是Spring 帮我们做,有的是靠Configuration来做,有的是靠注解来做而已。