使用SpringAMQP实现HelloWorld中的简单队列模型
基本消息队列(BasicQueue):
一、什么是 AMQP
可以参考:SpringAmqp的官方地址
Spring AMQP 的特性:
1.Listener container for asynchronous processing of inbound messages(用于异步处理入站消息的侦听器容器)
2.RabbitTemplat
for sending and receiving messages(RabbitTemplate用于发送和接收消息)
3.RabbitAdmin
for automatically declaring queues, exchanges and bindings(RabbitAdmin用于自动声明队列、交换和绑定)
二、引入依赖
应该在父工程中引入,因为无论是消息发送者还是接收者都要用这个依赖,由 Spring 为我们管理
<!--AMQP依赖,包含RabbitMQ-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-amqp</artifactId>
</dependency>
三、发消息
在 publisher 服务中利用 RabbitTemplate 发送消息到 simple.queue 这个队列
1.编写配置信息
在 publisher 服务中编写 application.yml,添加 mq 连接信息:
spring:
rabbitmq:
host: 192.168.3.128 # rabbitmq 的 ip 地址
port: 5672 # 端口号
virtual-host: / # 该用户使用的虚拟主机
username: gentlebrother # 用户名
password: gentlebrother # 密码
2.使用RabbitTemplate发消息
在 publisher 服务中编写测试类 SpringAmqpTest,并利用 RabbitTemplate 实现消息发送:
package cn.itcast.mq.spring;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;
/**
* @author 温柔哥
* @create 2024-02-01 11:44
*/
@SpringBootTest
@RunWith(SpringRunner.class)
public class SpringAmqpTest {
@Autowired
private RabbitTemplate rabbitTemplate;
@Test
public void testSendMessage2SimpleQueue() {
String queueName = "simple.queue";
String message = "hello rabbitmq";
rabbitTemplate.convertAndSend(queueName, message);
}
}
3.运行后报错
Caused by: java.nio.charset.MalformedInputException: Input length = 1
at java.nio.charset.CoderResult.throwException(CoderResult.java:281)
at sun.nio.cs.StreamDecoder.implRead(StreamDecoder.java:339)
at sun.nio.cs.StreamDecoder.read(StreamDecoder.java:178)
at java.io.InputStreamReader.read(InputStreamReader.java:184)
at org.yaml.snakeyaml.reader.UnicodeReader.read(UnicodeReader.java:125)
at org.yaml.snakeyaml.reader.StreamReader.update(StreamReader.java:183)
... 71 more
这是因为配置文件里面有中文或者是编码格式不正确导致
1.修改编码格式:
2.将 yml 文件中文注释删除掉,运行成功,到 rabbitmq 客户端查看消息
3.然后再将中文注释加上,再次运行仍然可以发送成功,可以看到已经两条消息了
四、消费消息
在 consumer 中编写消费逻辑,监听 simple.queue
1.编写配置信息
在 consumer 服务中编写 application.yml,添加 mq 连接信息:
spring:
rabbitmq:
host: 192.168.3.128 # rabbitmq 的 ip 地址
port: 5672 # 端口号
virtual-host: / # 该用户使用的虚拟主机
username: gentlebrother # 用户名
password: gentlebrother # 密码
2.使用RabbitListener监听消息
consumer 服务中新建一个类,编写消费逻辑:
package cn.itcast.mq.listener;
import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.stereotype.Component;
/**
* @author 温柔哥
* @create 2024-02-01 16:01
*/
@Component
public class SpringRabbitListener {
@RabbitListener(queues = "simple.queue")
public void listenSimpleQueue(String message) {
System.out.println("消费者接受到 simple.queue 的消息:【" + message + "】");
}
}
3.启动 Spring 消费消息
因为上述的代码是一个 bean,由 Spring 来管理的,也就是说接收消息是由 Spring 来处理的,所以必须把整个 Spring 运行起来。
成功消费两条消息
回到 rabbitmq 客户端查看,确实没有消息了
五、总结
1.什么是AMQP?
应用间消息通信的一种协议,与语言和平台无关。
2.SpringAMQP如何发送消息?
引入amqp的starter依赖
配置RabbitMQ地址
利用RabbitTemplate的convertAndSend方法
3.SpringAMQP如何接收消息?
引入amqp的starter依赖
配置RabbitMQ地址
定义类,添加@Component注解
类中声明方法,添加@RabbitListener注解,方法参数就时消息
注意:消息一旦消费就会从队列删除,RabbitMQ没有消息回溯功能