版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/lizc_lizc/article/details/80722090
RabbitMQ简介
AMQP,即Advanced Message Queuing Protocol,高级消息队列协议,是应用层协议的一个开放标准,为面向消息的中间件设计。消息中间件主要用于组件之间的解耦,消息的发送者无需知道消息使用者的存在,反之亦然。AMQP的主要特征是面向消息、队列、路由(包括点对点和发布/订阅)、可靠性、安全。
RabbitMQ是一个开源的AMQP实现,服务器端用Erlang语言编写,支持多种客户端,如:Python、Ruby、.NET、Java、JMS、C、PHP、ActionScript、XMPP、STOMP等,支持AJAX。用于在分布式系统中存储转发消息,在易用性、扩展性、高可用性等方面表现不俗。
RabbitMQ基本概念
1.Message
消息,消息是不具名的,它由消息头和消息体组成。消息体是不透明的,而消息头则由一系列的可选属性组成,这些属性包括routing-key(路由键)、priority(相对于其他消息的优先权)、delivery-mode(指出该消息可能需要持久性存储)等。
2.Publisher
消息的生产者,也是一个向交换器发布消息的客户端应用程序。
3.Exchange
交换器,用来接收生产者发送的消息并将这些消息路由给服务器中的队列。
4.Binding
绑定,用于消息队列和交换器之间的关联。一个绑定就是基于路由键将交换器和消息队列连接起来的路由规则,所以可以将交换器理解成一个由绑定构成的路由表。
5.Queue
消息队列,用来保存消息直到发送给消费者。它是消息的容器,也是消息的终点。一个消息可投入一个或多个队列。消息一直在队列里面,等待消费者连接到这个队列将其取走。
6.Connection
网络连接,比如一个TCP连接。
7.Channel
信道,多路复用连接中的一条独立的双向数据流通道。信道是建立在真实的TCP连接内地虚拟连接,AMQP 命令都是通过信道发出去的,不管是发布消息、订阅队列还是接收消息,这些动作都是通过信道完成。因为对于操作系统来说建立和销毁 TCP 都是非常昂贵的开销,所以引入了信道的概念,以复用一条 TCP 连接。
8.Consumer
消息的消费者,表示一个从消息队列中取得消息的客户端应用程序。
9.Virtual Host
虚拟主机,表示一批交换器、消息队列和相关对象。虚拟主机是共享相同的身份认证和加密环境的独立服务器域。每个 vhost 本质上就是一个 mini 版的 RabbitMQ 服务器,拥有自己的队列、交换器、绑定和权限机制。vhost 是 AMQP 概念的基础,必须在连接时指定,RabbitMQ 默认的 vhost 是 / 。
10.Broker
表示消息队列服务器实体。
Exchange 类型
Exchange分发消息时根据类型的不同分发策略有区别,目前共四种类型:direct、fanout、topic、headers 。下面只讲前三种模式。
1.Direct模式
消息中的路由键(routing key)如果和 Binding 中的 binding key 一致, 交换器就将消息发到对应的队列中。路由键与队列名完全匹配
2.Topic模式
topic 交换器通过模式匹配分配消息的路由键属性,将路由键和某个模式进行匹配,此时队列需要绑定到一个模式上。它将路由键和绑定键的字符串切分成单词,这些单词之间用点隔开。它同样也会识别两个通配符:符号“#”和符号“*”。#匹配0个或多个单词,*匹配一个单词。
3.Fanout模式
每个发到 fanout 类型交换器的消息都会分到所有绑定的队列上去。fanout 交换器不处理路由键,只是简单的将队列绑定到交换器上,每个发送到交换器的消息都会被转发到与该交换器绑定的所有队列上。很像子网广播,每台子网内的主机都获得了一份复制的消息。fanout 类型转发消息是最快的。
SpringBoot整合RabbitMQ
在pom.xml
中添加spring-boot-starter-amqp
的依赖
<dependencies>
<!-- 结合注解@ConfigurationProperties 将bean里面的属性和yml中的属性自动绑定 -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-configuration-processor</artifactId>
<optional>true</optional>
</dependency>
<!--RabbitMQ消息队列-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-amqp</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
在application.yml
中配置rabbitmq
相关内容
spring:
rabbitmq:
host: localhost
port: 5672
username: guest
password: guest
使用Direct模式
1.配置队列
package com.jeesite.modules.zh.rabbitmq.Direct;
import org.springframework.amqp.core.Queue;
import org.springframework.amqp.rabbit.connection.CachingConnectionFactory;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.config.ConfigurableBeanFactory;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import javax.annotation.Resource;
import org.springframework.amqp.rabbit.connection.ConnectionFactory;
import org.springframework.context.annotation.Scope;
/**
* 第一步
* 配置队列
*/
@Configuration
public class RabbitMQConfig {
static final String QUEUE = "direct_queue";
static final String MIAOSHA = "miao_sha";
@Resource
private RabbitConstants rabbitConstants;
/**
* Direct模式
* @return
*/
@Bean
public Queue directQueue() {
// 第一个参数是队列名字, 第二个参数是指是否持久化
return new Queue(QUEUE, true);
}
@Bean Queue miaoSha(){
// 第一个参数是队列名字, 第二个参数是指是否持久化
return new Queue(MIAOSHA, true);
}
}
2.创建一个User实体类
package com.lzc.rabbitmq.dataobject;
import lombok.Data;
import java.io.Serializable;
@Data
public class User implements Serializable {
private static final long serialVersionUID = -1262627851741431084L;
private String userId;
private String name;
}
3.接收者
package com.jeesite.modules.zh.rabbitmq.Direct;
import com.jeesite.common.mapper.JsonMapper;
import com.jeesite.modules.zh.dao.ZhMsActivityDao;
import com.jeesite.modules.zh.entity.ZhMsActivity;
import lombok.extern.slf4j.Slf4j;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
/**
* 第三步
* 接收者
*/
@Component
@Slf4j
public class Receiver {
@Autowired
ZhMsActivityDao zhMsActivityDao;
private static Logger logger = LoggerFactory.getLogger(JsonMapper.class);
// queues是指要监听的队列的名字
@RabbitListener(queues = RabbitMQConfig.QUEUE)
public void receiverDirectQueue(User user) {
logger.info("【receiverDirectQueue监听到消息】"+user.toString());
}
//miao_sha是指要监听的队列名字
@RabbitListener(queues = RabbitMQConfig.MIAOSHA)
public void receiverDirectQueueMiaoShao(String str) {
ZhMsActivity ms = new ZhMsActivity();
ms.setId("1175976705438756864");
ms = zhMsActivityDao.get(ms);
if(ms.getSyNum()>0){
ms.setSyNum(ms.getSyNum()-1);
zhMsActivityDao.update(ms);
logger.info("【receiverDirectQueueMiaoShao】"+str+"-----抢单成功");
}else{
logger.info("【receiverDirectQueueMiaoShao】"+str+"-----抢单失败");
}
}
}
4.发送者
package com.jeesite.modules.zh.rabbitmq.Direct;
import com.jeesite.common.idgen.IdGen;
import com.jeesite.modules.zh.dao.ZhMsActivityDao;
import com.jeesite.modules.zh.entity.ZhMsActivity;
import com.jeesite.modules.zh.utils.AjaxBeanUtil;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
/**
* 第五步
* 测试,访问http://localhost:9999/f/sendDirectQueue,查看日志输出
*/
@RestController
@RequestMapping(value = "${frontPath}/direct/")
public class DirectTestController {
@Autowired
private Sender sender;
@Autowired
ZhMsActivityDao zhMsActivityDao;
@GetMapping("/sendDirectQueue")
public Object sendDirectQueue(){
sender.sendDirectQueue();
return "ok";
}
@GetMapping("/sendPhone")
public Object sendPhone(){
ZhMsActivity ms = new ZhMsActivity();
ms.setId("1175976705438756864");
ms = zhMsActivityDao.get(ms);
if(ms.getSyNum()>0){
sender.sendDirectMiaoShaPhone("模拟用户"+IdGen.nextId());
return "ok";
}else {
return "false";
}
}
}
6.日志输出
2018-06-18 01:18:54.901 INFO 8772 --- [io-8080-exec-10] com.lzc.rabbitmq.config.Sender : 【sendDirectQueue已发送消息】
2018-06-18 01:18:54.997 INFO 8772 --- [cTaskExecutor-1] com.lzc.rabbitmq.config.Receiver : 【receiverDirectQueue监听到消息】User(userId=123456, name=lizhencheng)