一、概述
- 大多应用中,可通过消息服务中间件来提升系统异步通信、扩展解耦能力
- 消息服务中两个重要概念:消息代理 (message broker) 和目的地 (destination)。
当消息发送者发送消息以后,将由消息代理接管,消息代理保证消息传递到指定目的地。 - 消息队列主要有两种形式的目的地:
①队列 (queue):点对点消息通信 (point-to-point)
②主题 (topic):发布 (publish) /阅 (subscribe) 消息通信 - 点对点式
①消息发送者发送消息,消息代理将其放入一个队列中,消息接收者从队列中获取消息内容,消息读取后被移出队列。
②消息只有唯一的发送者和接受者,但并不是说只能有一个接收者 - 发布订阅式:
发送者(发布者) 发送消息到主题,多个接收者 (订阅者)监听(订阅)这个主题,那么就会在消息到达时同时收到消息 - JMS(Java Message Service) JAVA消息服务
基于JVM消息代理的规范。ActiveMQ、HornetMQ是JMS实现 - AMQP(Advanced Message Queuing Protocol)
①高级消息队列协议,也是一个消息代理的规范,兼容JMS
②RabbitMQ是AMQP的实现
JMS | AMQP | |
---|---|---|
定义 | Java Api | 网络级协议 |
跨语言 | 否 | 是 |
跨平台 | 否 | 是 |
Model | 提供两种消息模型:1、Peer-2-Peer 2、Pub/sub | 提供了五种消息模型:1、direct exchange 2、fanout exchange 3、topic change 4、headers exchange 5、system exchange。本质来讲,后四种和JMS的pub/sub模型没有太大差别,仅是在路由机制上做了更详细的划分 |
支持消息类型 | 多种消息类型:TextMessage、MapMessage、BytesMessage、StreamMessage、ObjectMessage、Message(只有消息头和属性) | byte[];当实际应用时,有复杂的消息,可以将消息序列化后发送。 |
综合评价 | JMS定义了JAVA API层面的标准;在java体系中,多个clien均可以通过JMS进行交互,不需要应用修改代码,但是其对跨言特性。平台的支持较差 | AMQP定义了wire-level层的协议标准;天然具有跨平台、跨语言特性。 |
二、架构
三、Exchange(交换器)类型
目前有四种:direct、 fanout、 topic、headers
(1)direct 路由键(routing key)与队列名(name)完全一致
(2)fanout 交换器下所有队列都接收到消息
(3)topic 将路由键和某个模式进行匹配 ; #匹配0个或多个单词,*匹配一个单词
三、安装RabbitMQ
-
安装erlang,下载地址:http://www.erlang.org/downloads,双击.exe文件进行安装就好。安装完成之后将erlang的安装目录(C:\ProgramFiles\erl9.1\bin,具体请查看安装路径)加入到Path中。最后打开命令行,输入erl,如果出现erlang的版本信息就表示erlang语言环境安装成功;
-
安装RabbitMQ,下载地址:http://www.rabbitmq.com/download.html,同样双击.exe进行安装。修改RabbitMQ安装目录为 D:\SinoWeb\RabbitMQ;
-
安装RabbitMQ-Plugins
安装方法是:打开命令行cd进入rabbitmq的sbin目录(D:\SinoWeb\RabbitMQ\rabbitmq_server-3.6.12\sbin),输入一下安装命令
安装命令:
rabbitmq-plugins enable rabbitmq_management
开通STOMP通道:
rabbitmq-plugins enable rabbitmq_web_stomp
rabbitmq-plugins enable rabbitmq_web_stomp_examples -
插件安装完之后,双击rabbitmq-server.bat,在浏览器输入http://localhost:15672进行验证,你会看到登陆界面,输入用户名:guest,密码:guest你就可以进入管理界面;
-
创建用户
使用guest账户登陆,点击Admin标签,然后点击 Add a user,在新增用户的表单中输入以下内容
Username:sino
Password:sino3210
Tags: administrator
点击Add User按钮创建用户。
创建完毕后 新创建的用户会显示No access。点击刚才创建的sino用户名,在新页面中, 点击Set permission按钮。
四、使用RabbitMQ
(1)点击Exchanges按钮
(2)点击Quene按钮
(3)点击Exchange,绑定quene
五、SpringBoot整合
1、引入依赖
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-amqp</artifactId>
</dependency>
1、配置文件添加属性
spring:
rabbitmq:
host: 127.0.0.1
port: 5672
username: guest
password: guest
2、代码
@Autowired
private RabbitTemplate rabbitTemplate;
@Autowired
private AmqpAdmin amqpAdmin;
//发送和接收信息
@Test
void contextLoads() {
//1、direct类型交换器,单个routingKey ----SUCCESS
rabbitTemplate.convertAndSend("text.exchange", "text.quene",
"我是发送到text.exchange交换器,text.quene消息队列的信息");
//2、fauout类型交换器,routingKey不传参数 -----SUCCESS
rabbitTemplate.convertAndSend("text.exchange2", "",
"我是发送到text.exchange2交换器,fauout消息队列的信息");
//3、fauout类型交换器,不管是否传参routingKey,都发送到交换器下所以的队列 ----ERROR
rabbitTemplate.convertAndSend("text.exchange2", "text.quene",
"我是发送到text.exchange2交换器,text.quene单独测试fauout消息队列的信息");
//4、direct类型交换器,routingKey不传参数,所以的队列都不接收信息 ----ERROR
rabbitTemplate.convertAndSend("text.exchange", "",
"我是发送到text.exchange交换器,所以队列信息");
//5、接收信息
Object convert = rabbitTemplate.receiveAndConvert("text4.quene");
System.err.println(convert);
}
备注:当传输的消息是Map类型或实体类时,RabbitMQ接收的消息是转化乱码的,需转成json格式
添加 MyAmqpConfig 配置类
@Test
void contextLoads() {
Map<String, Object> map = new HashMap<>();
map.put("msg", "这是第二个消息");
map.put("data", Arrays.asList("HelloWorld", "小米", new Date()));
rabbitTemplate.convertAndSend("test.direct", "five.quene", map);
}
@Configuration
public class MyAmqpConfig {
@Bean
public MessageConverter messageConverter(){
return new Jackson2JsonMessageConverter();
}
}
3、使用@RabbitListener监听消息队列的内容
(1)@EnableRabbit注解开启RabbitMQ
(2)在方法上标注@RabbitListener
@EnableRabbit
@SpringBootApplication
public class DemoApplication {
public static void main(String[] args) {
SpringApplication.run(DemoApplication.class, args);
}
}
@Configuration
public class MyRabbitConfig {
@Bean
public MessageConverter messageConverter() {
return new Jackson2JsonMessageConverter();
}
}
@Service
public class ReceiveService {
@RabbitListener(queues = "text.quene")
public void receive(String message){
System.err.println(message);
System.err.println(message.getBody());
}
}
4、AmqpAdmin
创建和删除Quene、Exchange、Bind
@Autowire
AmqpAdmin amqpAdmin;
//创建交换器和队列
@Test
public void createExchange(){
//创建交换器
amqpAdmin.declareExchange(new DirectExchange("rabDirect.exchange"));
amqpAdmin.declareExchange(new FanoutExchange("rabFanout.exchange"));
//创建队列
amqpAdmin.declareQueue(new Queue("allen.quene", true));
//绑定队列
amqpAdmin.declareBinding(new Binding("allen.quene",
Binding.DestinationType.QUEUE, "rabFanout.exchange",
"allen.quene", null));
amqpAdmin.declareBinding(new Binding("allen.quene",
Binding.DestinationType.QUEUE, "rabDirect.exchange",
"allen.quene", null));
}