Java客户端
快速入门
将来我们开发业务功能的时候,肯定不会在控制台收发消息,而是应该基于编程的方式。
data:image/s3,"s3://crabby-images/181c2/181c29670c6857a4c278c53995e25b8e2836e3a5" alt=""
- 由于
RabbitMQ
采用了AMQP协议,因此它具备跨语言的特性。任何语言只要遵循AMQP协议收发消息,都可以与RabbitMQ
交互。并且RabbitMQ
官方也提供了各种不同语言的客户端。 - 但是,RabbitMQ官方提供的Java客户端编码相对复杂,一般生产环境下我们更多会结合Spring来使用。
- 而Spring的官方刚好基于RabbitMQ提供了这样一套消息收发的模板工具:SpringAMQP。并且还基于SpringBoot对其实现了自动装配,使用起来非常方便。
在课前资料给大家提供了一个Demo工程,方便我们学习SpringAMQP的使用:
data:image/s3,"s3://crabby-images/fc775/fc77567b5b085775cc6a121ecb8dc8c7ba70a0f6" alt=""
data:image/s3,"s3://crabby-images/af390/af3902d4b69a6dc70e89f90aba978b29b443421a" alt=""
- mq-demo:父工程,管理项目依赖
- publisher:消息的发送者
- consumer:消息的消费者
案例需求: 利用SpringAMQP完成消息的收发
在之前的案例中,我们都是经过交换机发送消息到队列,不过有时候为了测试方便,我们也可以直接向队列发送消息,跳过交换机。这种模式一般测试使用,很少在生产中使用。
在入门案例中,我们就演示这样的简单模型,如图:
data:image/s3,"s3://crabby-images/51143/51143297aee97b651821fc3a053fb9124393c515" alt=""
也就是:
- publisher直接发送消息到队列
- 消费者监听并处理队列中的消息
-
在父工程中引入spring-amqp依赖,这样publisher和consumer服务都可以使用
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-amqp</artifactId> </dependency> -
配置rabbitmq服务端信息, 微服务才能连接到RabbitMO
spring:
rabbitmq:
host: 192.168.0.105 # 你的虚拟机IP
port: 5672 # 端口
virtual-host: /hmall # 虚拟主机
username: hmall # 用户名
password: 123 # 密码 -
利用控制台创建队列simple.queue
data:image/s3,"s3://crabby-images/bd73f/bd73f9bfed8218a9742871979a707b384f4460e5" alt=""
- 之前测试数据隔离时, 账号和队列都已经建好了
- 账密: hmall/123
- 发送消息: 在publisher服务中,利用SpringAMQP提供的RabbitTemplate工具类, 发送信息
data:image/s3,"s3://crabby-images/a3d6a/a3d6a03dc87d9e478da546bd21998bbc622588fc" alt=""
@SpringBootTest
public class SpringAmqpTest {
@Autowired
private RabbitTemplate rabbitTemplate;
@Test
public void testSimpleQueue() {
//1.队列名
String queueName = "simple.queue";
//2.消息
String message = "Hello,Spring AMQP!";
//3.发送消息
rabbitTemplate.convertAndSend(queueName, message);
}
}
- 监听队列: SpringAMQP提供声明式的消息监听,通过注解就可以把消息传递给当前方法
data:image/s3,"s3://crabby-images/39a78/39a78a2077d132a2362df698c00215b6fc21ff43" alt=""
- 消息发送时是什么类型, 接收时就用同类型接收, spring会自动处理
- 测试一下
- 启动publisher(发送者)发送消息
data:image/s3,"s3://crabby-images/db79f/db79f1d1aff89114531df7eeead9db2a70404b26" alt=""
- 启动consumer(消费者)接受消息
data:image/s3,"s3://crabby-images/96b43/96b43bad8dab83de479db00d7f5a56f95821ca4e" alt=""
WorkQueue
Work queues,任务模型。简单来说就是让多个消费者绑定到一个队列,共同消费队列中的消息
data:image/s3,"s3://crabby-images/f8e4a/f8e4a9f3e58bbb11b85ae4622301180f9c84d645" alt=""
模拟WorkQueue,实现一个队列绑定多个消费者
基本思路如下:
- 在RabbitMQ的控制台创建一个队列,名为work.queue
data:image/s3,"s3://crabby-images/6f680/6f680cc93ef042b3caca8c6142094b2adc68b8f2" alt=""
-
在publisher服务中定义测试方法,发送50条消息到work.queue
@SpringBootTest
public class SpringAmqpTest {@Autowired private RabbitTemplate rabbitTemplate; @Test public void testWorkQueue() { //1.队列名 String queueName = "work.queue"; for (int i = 1; i < 50; i++) { //2.消息 String message = "Hello,Spring AMQP_" + i; //3.发送消息 rabbitTemplate.convertAndSend(queueName, message); } }
}
-
在consumer服务中定义两个消息监听者,都监听work.queue队列
@Slf4j
@Component
public class SpringRabbitListener {@RabbitListener(queues = "work.queue") public void listenWorkQueue1(String msg) throws InterruptedException{ System.out.println("消费者1收到消息:" + msg + "," + LocalTime.now()); } @RabbitListener(queues = "work.queue") public void listenWorkQueue2(String msg) throws InterruptedException{ System.err.println("消费者2...收到消息:" + msg + "," + LocalTime.now()); }
}
-
重启代码, 查看执行结果
data:image/s3,"s3://crabby-images/9faaf/9faaf3dde35f49d65cfb8fd9791e2d15c74972fd" alt=""
data:image/s3,"s3://crabby-images/0425a/0425ae99ca3be1ee1b3239699bc0bee2aed907be" alt=""
- 默认情况下,RabbitMQ的会将消息依次轮询投递给绑定在队列上的每一个消费者。
- 但这并没有考虑到消费者是否已经处理完消息,可能出现消息堆积。
- 因此我们需要修改application.yml,设置preFetch值为1,确保同一时刻最多投递给消费者1条消息
- 让消费者1每秒处理40条消息,消费者2每秒处理5条消息
- 配置按需获取消息
data:image/s3,"s3://crabby-images/815ae/815ae77b507dd7cf9ddd1833bf5c92a1907504fa" alt=""
- 模拟消息处理能力差异
data:image/s3,"s3://crabby-images/e5731/e5731bfea625d734681cb4f1d1ae9a4780389ad8" alt=""
- 重启测试: 处理能力越强, 处理的消息越多, 避免了消息堆积, 影响效率
data:image/s3,"s3://crabby-images/ad496/ad49649d6939b9aaafcef4fe1d9b57352f6c01b3" alt=""
Work模型的使用
- 多个消费者绑定到一个队列,可以加快消息处理速度
- 同一条消息只会被一个消费者处理
- 通过设置prefetch来控制消费者预取的消息数量,处理完一条再处理下一条,实现能者多劳
Fanout交换机
交换机的作用主要是接收发送者发送的消息,并将消息路由到与其绑定的队列。
data:image/s3,"s3://crabby-images/8d9fb/8d9fb88343173ac0da20b85d92ad591a2ec18da1" alt=""
常见交换机的类型:
- Fanout: 广播
- Direct: 定向
- Topic: 话题
Fanout Exchange 会将接收到的消息路由到每一个跟其绑定的queue,所以也叫广播模式
data:image/s3,"s3://crabby-images/53559/53559c8780145d4e746849451e19949959edcf01" alt=""
利用SpringAMQP演示FanoutExchange的使用
data:image/s3,"s3://crabby-images/bc979/bc9792eddec529a8258245a9e07896b53af7966a" alt=""
- 在RabbitMQ控制台中,声明队列fanout.queue1和fanout.queue2
data:image/s3,"s3://crabby-images/5a34f/5a34fa15b4a0990f73a620109607efb309caa350" alt=""
- 账号: hmall/123
- 在RabbitMO控制台中,声明交换机hmall.fanout,将两个队列与其绑定
data:image/s3,"s3://crabby-images/30522/305220caedb669862e4ec4ffa4d554c8504365c6" alt=""
data:image/s3,"s3://crabby-images/bc4bf/bc4bf0385cb2b7e60297fc0bd9ad6548fb01ddda" alt=""
data:image/s3,"s3://crabby-images/8ca07/8ca07fd41e591c7df591b59ab1a5d2fac5c10c98" alt=""
- 在consumer服务中,编写两个消费者方法,分别监听fanout.queue1和fanout.queue2
data:image/s3,"s3://crabby-images/f2dac/f2dac283f6a3c1f48df0ed7825ccad29c5856606" alt=""
@Slf4j
@Component
public class SpringRabbitListener {
@RabbitListener(queues = "fanout.queue1")
public void listenFanoutQueue1(String msg) {
log.info("消费者1监听到 fanout.queue1的消息:{}", msg);
}
@RabbitListener(queues = "fanout.queue2")
public void listenFanoutQueue2(String msg) {
log.info("消费者2监听到 fanout.queue2的消息:{}", msg);
}
}
- 在publisher中编写测试方法,向hmall.fanout发送消息
data:image/s3,"s3://crabby-images/962ff/962ffd62aa083c4747e97f049b2e7db8cc825d46" alt=""
@SpringBootTest
public class SpringAmqpTest {
@Autowired
private RabbitTemplate rabbitTemplate;
@Test
public void testFanoutQueue() {
//1.交换机名称
String exchangeName = "hmall.fanout";
//2.消息
String message = "Hello, everyone";
//3.发送消息 参数: 交换机名称, RoutingKey(暂时未空), 消息
rabbitTemplate.convertAndSend(exchangeName, null, message);
}
}
- 启动服务, 使用publisher服务发送消息, 在consumer服务监听消息
data:image/s3,"s3://crabby-images/8b25d/8b25d61e1ca01017d4a421d9eb183a7c8222d658" alt=""
Direct交换机
Direct Exchange 会将接收到的消息根据规则路由到指定的Queue,因此称为定向路由。
data:image/s3,"s3://crabby-images/4f751/4f7513588dfcb296c9b56a8e81ee4e0415e22359" alt=""
- 每一个Queue都与Exchange设置一个BindingKey。
- 发布者发送消息时,指定消息的RoutingKey
- Exchange将消息路由到BindingKey与消息RoutingKey一致的队列
利用SpringAMQP演示DirectExchange的使用
- 在RabbitMQ控制台中,声明队列direct.queue1和direct.quque2
data:image/s3,"s3://crabby-images/5ae5e/5ae5ea828bd133c389a4e6df6ee37074a057e84b" alt=""
- 在RabbitMO控制台中,声明交换机hmall.direct,将两个队列与其绑定
data:image/s3,"s3://crabby-images/dff7a/dff7a07292dd3022b57a6f80ed999e0905d0bf37" alt=""
data:image/s3,"s3://crabby-images/64d3a/64d3a3bf2020727168705cadf61ce47fa8b3290a" alt=""
-
在consumer服务中,编写两个消费者方法,分别监听direct.queue1和direct.queue2
@Slf4j
@Component
public class SpringRabbitListener {@RabbitListener(queues = "direct.queue1") public void listenDirectQueue1(String msg) { log.info("消费者1监听到 direct.queue1的消息:{}", msg); } @RabbitListener(queues = "direct.queue2") public void listenDirectQueue2(String msg) { log.info("消费者2监听到 direct.queue2的消息:{}", msg); }
}
-
在publisher中编写测试方法,利用不同的RoutingKey向hmall.direct发送消息
@SpringBootTest
public class SpringAmqpTest {@Autowired private RabbitTemplate rabbitTemplate; @Test public void testDirectQueue() { //1.队列名 String exchangeName = "hmall.direct"; //2.消息 String message = "红色: 震惊, 居然卡爆了"; //3.发送消息 参数: 交换机名称, RoutingKey, 消息 rabbitTemplate.convertAndSend(exchangeName, "red", message); }
}
-
启动服务, 发消息测试
data:image/s3,"s3://crabby-images/a4822/a48225de73376fd916a181a8915f77c01a52920a" alt=""
data:image/s3,"s3://crabby-images/4e209/4e2094c5b6bd87435e5034734966c4a3224604ff" alt=""
描述下Direct交换机与Fanout交换机的差异?
- Fanout交换机将消息路由给每一个与之绑定的队列
- Direct交换机根据RoutingKey判断路由给哪个队列
- 如果多个队列具有相同RoutingKey,则与Fanout功能类似
Topic交换机
TopicExchange也是基于RoutingKey做消息路由,但是routingKey通常是多个单词的组合,并且以 . 分割
data:image/s3,"s3://crabby-images/e7a95/e7a956b06fd319c412b96f8048abdf6b6b062aff" alt=""
Topic交换机与队列绑定时, BindingKey可以使用通配符:
#: 代指0个或多个单词
*: 代指一个单词
利用SpringAMQP演示DirectExchange的使用
data:image/s3,"s3://crabby-images/b7358/b7358e61834f949b6dc9c5a32c7d9e3470114587" alt=""
- 在RabbitMQ控制台中,声明队列topic.queue1和topic.queue2
data:image/s3,"s3://crabby-images/8aaf0/8aaf0bf7ee7230fe3f92b6ac1849d1300c62673c" alt=""
- 在RabbitMQ控制台中,声明交换机hmall.topic,将两个队列与其绑定
data:image/s3,"s3://crabby-images/3127f/3127f209465caac83c01106da6edc3fae9c48de8" alt=""
data:image/s3,"s3://crabby-images/68272/6827248a9a9a0662cde87d9ddc461d33cfa635b7" alt=""
-
在consumer服务中,编写两个消费者方法,分别监听topic.queue1和topic.queue2
@Slf4j
@Component
public class SpringRabbitListener {@RabbitListener(queues = "topic.queue1") public void listenTopicQueue1(String msg) { log.info("消费者1监听到 topic.queue1的消息:{}", msg); } @RabbitListener(queues = "topic.queue2") public void listenTopicQueue2(String msg) { log.info("消费者2监听到 topic.queue2的消息:{}", msg); }
}
-
在publisher中编写测试方法,利用不同的RoutingKey向hmall.topic发送消息
@SpringBootTest
public class SpringAmqpTest {@Autowired private RabbitTemplate rabbitTemplate; @Test public void testTopicQueue() { //1.队列名 String exchangeName = "hmall.topic"; //2.消息 String message = "都能收到的消息"; //3.发送消息 参数: 交换机名称, RoutingKey, 消息
// rabbitTemplate.convertAndSend(exchangeName, "china.news", message);
// rabbitTemplate.convertAndSend(exchangeName, "china.info", message);
rabbitTemplate.convertAndSend(exchangeName, "test.news", message);
}}
-
启动服务进行测试: 根据不同RoutingKey, 交换机把消息分发给不同的消费者
data:image/s3,"s3://crabby-images/24276/24276243791fbc0dc300a6d26648d4aef605a217" alt=""
描述下Topic交换机相比Direct交换机的优势
- 实现的功能时类似的
- 在绑定bindingKey时可以使用通配符, 扩展性和灵活性更好
声明队列交换机
前面我们使用过web控制台的方式创建交换机和队列, 并进行绑定, 实际工作中, 需要使用java代码自动创建
SpringAMQP提供了几个类,用来声明队列、交换机及其绑定关系:
- SpringAMQP提供了Queue类, 用于声明队列
data:image/s3,"s3://crabby-images/071d2/071d2c93776c00b950b0853fa960ab0a627ddb52" alt=""
- SpringAMQP提供了Exchange接口,来表示所有不同类型的交换机:
data:image/s3,"s3://crabby-images/09964/099641200bfdeea9a36c3bc56deb0b6c92bdd713" alt=""
- 我们可以自己创建队列和交换机,SpringAMQP还提供了ExchangeBuilder来简化这个过程
data:image/s3,"s3://crabby-images/0ef8a/0ef8aa2bf63da8d8f9898ab9d84857e9d61acd52" alt=""
- Binding类: 用于声明队列和交换机的绑定关系,可以用工厂类BindingBuilder构建
data:image/s3,"s3://crabby-images/c5a82/c5a827bf431bebfec5362746a2a272385a172a0a" alt=""
fanout示例
- 先删掉前面创建的hmall.fanout交换机, 以及队列fanout.queue1/fanout.queue2
- 在consumer中创建一个类, 声明一个Fanout类型的交换机,并且创建队列与其绑定
- 创建工作一般放在消费者服务, 因为消息发送者只负责发消息,
- 消费者接收消息就要考虑交换机和队列, 以及绑定关系
data:image/s3,"s3://crabby-images/79f50/79f506c314866462afa1c531b18d1b042e0e269d" alt=""
@Configuration
public class FanoutConfiguration {
// 声明FanoutExchange交换机
@Bean
public FanoutExchange fanoutExchange() {
// 使用构建器创建
// return ExchangeBuilder.fanoutExchange("hmall.fanout").build();
// 手动创建
return new FanoutExchange("hmall.fanout");
}
// 声明队列1
@Bean
public Queue fanoutQueue1() {
// 使用构建器创建
// return QueueBuilder.durable("fanout.queue1").build();
// 手动创建
return new Queue("fanout.queue1");
}
// 绑定队列1和交换机
@Bean
public Binding bindingQueue1(Queue fanoutQueue1, FanoutExchange fanoutExchange) {
return BindingBuilder.bind(fanoutQueue1).to(fanoutExchange);
}
// 声明队列2
@Bean
public Queue fanoutQueue2() {
return new Queue("fanout.queue2");
}
// 绑定队列1和交换机
@Bean
public Binding bindingQueue2(Queue fanoutQueue2, FanoutExchange fanoutExchange) {
return BindingBuilder.bind(fanoutQueue2).to(fanoutExchange);
}
}
- 重启服务: 队列和交换机已经自动创建, 绑定关系也有
data:image/s3,"s3://crabby-images/d4815/d48158b8d42d4f97a7e83146c9f383808700881e" alt=""
data:image/s3,"s3://crabby-images/fe8da/fe8da89a7311c5580765f085b03584d5fc37b396" alt=""
data:image/s3,"s3://crabby-images/968af/968af70a57fd7f3cac46fa8537da709d95dc33c7" alt=""
direct示例
- 先删掉前面创建的hmall.direct交换机, 以及队列direct.queue1/direct.queue2
- 利用SpringAMQP声明DirectExchange并与队列绑定
- direct模式由于要绑定多个KEY,会非常麻烦,每一个Key都要编写一个binding
data:image/s3,"s3://crabby-images/a6119/a61199d839a9096e96b5f88b870235d2c7fb98b7" alt=""
@Configuration
public class DirectConfiguration {
/**
* 声明交换机
* @return Direct类型交换机
*/
@Bean
public DirectExchange directExchange(){
return ExchangeBuilder.directExchange("hmall.direct").build();
}
/**
* 创建队列1
*/
@Bean
public Queue directQueue1(){
return new Queue("direct.queue1");
}
/**
* 绑定队列和交换机
*/
@Bean
public Binding bindingQueue1WithRed(Queue directQueue1, DirectExchange directExchange){
return BindingBuilder.bind(directQueue1).to(directExchange).with("red");
}
/**
* 绑定队列和交换机
*/
@Bean
public Binding bindingQueue1WithBlue(Queue directQueue1, DirectExchange directExchange){
return BindingBuilder.bind(directQueue1).to(directExchange).with("blue");
}
/**
* 创建队列2
*/
@Bean
public Queue directQueue2(){
return new Queue("direct.queue2");
}
/**
* 绑定队列和交换机
*/
@Bean
public Binding bindingQueue2WithRed(Queue directQueue2, DirectExchange directExchange){
return BindingBuilder.bind(directQueue2).to(directExchange).with("red");
}
/**
* 绑定队列和交换机
*/
@Bean
public Binding bindingQueue2WithYellow(Queue directQueue2, DirectExchange directExchange){
return BindingBuilder.bind(directQueue2).to(directExchange).with("yellow");
}
}
- 执行代码: 交换机, 队列以及绑定关系已经创建
data:image/s3,"s3://crabby-images/7d4b6/7d4b650ee8bd0609617249818430f8973241ef9c" alt=""
data:image/s3,"s3://crabby-images/51e35/51e35c2b7d3f267a6dd19ec6ac93e9e5828d96d6" alt=""
data:image/s3,"s3://crabby-images/70ace/70ace0ac409b28bf1aba2b6c5620179f802d8d54" alt=""
SpringAMQP还提供了基于@RabbitListener注解来声明队列和交换机的方式:
data:image/s3,"s3://crabby-images/884a8/884a8b81c0b826c6878ee605724c7eb1e64f19fc" alt=""
@Slf4j
@Component
public class SpringRabbitListener {
@RabbitListener(bindings = @QueueBinding(
value = @Queue(name = "direct.queue1"),
exchange = @Exchange(name = "hmall.direct", type = ExchangeTypes.DIRECT),
key = {"red", "blue"}
))
public void listenDirectQueue1(String msg) {
log.info("消费者1监听到 direct.queue1的消息:{}", msg);
}
@RabbitListener(bindings = @QueueBinding(
value = @Queue(name = "direct.queue2"),
exchange = @Exchange(name = "hmall.direct", type = ExchangeTypes.DIRECT),
key = {"red", "yellow"}
))
public void listenDirectQueue2(String msg) {
log.info("消费者2监听到 dirce.queue2的消息:{}", msg);
}
}
消息转换器
Spring的消息发送代码接收的消息体是一个Object:
data:image/s3,"s3://crabby-images/36849/3684902ba7ce2b4643485e6ac3a53f0ac5c68185" alt=""
在数据传输时,它会把发送的消息序列化为字节发送给MQ,接收消息的时候,还会把字节反序列化为Java对象。
只不过,默认情况下Spring采用的序列化方式是JDK序列化。众所周知,JDK序列化存在下列问题:
- 数据体积过大
- 有安全漏洞
- 可读性差
我们来测试一下。
- 声明一个队列,名为object.queue
data:image/s3,"s3://crabby-images/5af05/5af055b30e4c2036852594fd2a1d0c6752320dfa" alt=""
-
编写单元测试,向队列中直接发送一条消息, 消息类型为Map
@SpringBootTest
public class SpringAmqpTest {@Autowired private RabbitTemplate rabbitTemplate; @Test public void testSendObject() { //1.准备消息 Map<String, Object> msg = new HashMap<>(2); msg.put("name", "tom"); msg.put("age", 18); //2.发送消息 rabbitTemplate.convertAndSend("object.queue", msg); }
}
-
在控制台查看消息
data:image/s3,"s3://crabby-images/d5402/d5402aa0a480557ce77f2f9ad523f00282e0ee78" alt=""
- Spring的消息对象的处理默认是基于JDK的ObjectOutputStream完成序列化。存在明显的问题:
- ·JDK的序列化有安全风险
- JDK序列化的消息太大
- JDK序列化的消息可读性差
建议采用JSON序列化代替默认的JDK序列化,要做两件事情
- 在publisher和consumer中都要引入jackson依赖:
- 我们直接在父工程中引入, 避免重复引入
- 引入后刷新一下mave, 确保子工程加载依赖
data:image/s3,"s3://crabby-images/69f37/69f37e88c0025cc16b25bf7d4a243e1b1ae2f61a" alt=""
<dependency>
<groupId>com.fasterxml.jackson.dataformat</groupId>
<artifactId>jackson-dataformat-xml</artifactId>
<version>2.9.10</version>
</dependency>
-
配置消息转换器,在
publisher
和consumer
两个服务的启动类中添加一个Bean@SpringBootApplication
public class PublisherApplication {
public static void main(String[] args) {
SpringApplication.run(PublisherApplication.class);
}@Bean public MessageConverter messageConverter(){ return new Jackson2JsonMessageConverter(); }
}
-
消费者接收Object
-
在consumer服务中定义一个新的消费者
-
publisher是用Map发送,那么消费者也一定要用Map接收
@Slf4j
@Component
public class SpringRabbitListener {@RabbitListener(queues = "object.queue") public void listenObjectQueue(Map<String, Object> msg) { log.info("消费者监听到 Object.queue的消息:{}", msg); }
}
- 重启服务, 查看消息
data:image/s3,"s3://crabby-images/5d91e/5d91e35da0889c027ef21c28c464989aa0c0dbc7" alt=""
业务改造
需求 :改造余额支付功能,不再同步调用交易服务的OpenFeign接口,而是采用异步MQ通知交易服务更新订单状态
data:image/s3,"s3://crabby-images/5f32d/5f32d97a84ee8f07a051c02042c4a9b6d0f03b64" alt=""
data:image/s3,"s3://crabby-images/3b35e/3b35edf34ceae5f5eb2bf2889029350d0e9ba418" alt=""
- 配置MQ: 不管是生产者(pay-service) 还是 消费者(trade-service),都需要配置MQ的基本信息。
引入依赖
<!--消息发送-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-amqp</artifactId>
</dependency>
配置地址
spring:
rabbitmq:
host: 192.168.1.97 # 你的虚拟机IP
port: 5672 # 端口
virtual-host: /hmall # 虚拟主机
username: hmall # 用户名
password: 123 # 密码
配置消息转换器
- 因为多个服务都要使用mq, 所以每个服务都配置消息转换器太麻烦, 就在commom服务中进行配置
data:image/s3,"s3://crabby-images/b5f60/b5f60b8e0d22cd3ec493c1a78e1c41aeb85c69e3" alt=""
@Configuration
public class MqConfig {
@Bean
public MessageConverter messageConverter() {
return new Jackson2JsonMessageConverter();
}
}
- 配置类生效的前提是让spring扫描器扫描到, 在factories文件中指定文件, 让配置文件生效
data:image/s3,"s3://crabby-images/9cb54/9cb54fae20e56116a87e508eeaa557ad5498a56e" alt=""
org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
com.hmall.common.config.MyBatisConfig,\
com.hmall.common.config.MvcConfig,\
com.hmall.common.config.MqConfig,\
com.hmall.common.config.JsonConfig
- 接收消息: 在trade-service服务中定义一个消息监听类
data:image/s3,"s3://crabby-images/05523/055235163a915377213701c8f26b75bb453170ac" alt=""
@Component
@RequiredArgsConstructor
public class PayStatusListener {
private final IOrderService orderService;
@RabbitListener(bindings = @QueueBinding(
value = @Queue(value = "trade.pay.success.queue", durable = "true"),
exchange = @Exchange(name = "pay.direct"),
key = "pay.success"
))
public void listenPatSuccess(Long orderId) {
orderService.markOrderPaySuccess(orderId);
}
}
-
发送消息: 修改
pay-service
服务下的PayOrderServiceImpl
类中的tryPayOrderByBalance
方法@Slf4j
@Service
@RequiredArgsConstructor
public class PayOrderServiceImpl extends ServiceImpl<PayOrderMapper, PayOrder> implements IPayOrderService {private final RabbitTemplate rabbitTemplate; @Override @Transactional public void tryPayOrderByBalance(PayOrderFormDTO payOrderFormDTO) { ... ... // 5.修改订单状态 // tradeClient.markOrderPaySuccess(po.getBizOrderNo()); try { rabbitTemplate.convertAndSend("pay.direct", "pay.success", po.getBizOrderNo()); } catch (Exception e) { log.error("发送支付状态通知失败, 订单id: {}", po.getBizOrderNo(), e); } }
}
- 异步通知尽量不要对原有业务产生影响, 简单的处理就是使用try捕获异常
- 重启服务, 进行测试