RabbitMQ入门案例
RabbitMQ入门案例
Rabbit 模式
实现步骤
- 构建一个 maven工程
- 导入 rabbitmq的依赖
- 启动 rabbitmq-server服务
- 定义生产者
- 定义消费者
- 观察消息的在 rabbitmq-server服务中的进程
初步实现
前期准备
1.构建项目
2.导入依赖
<dependency>
<groupId>com.rabbitmq</groupId>
<artifactId>amqp-client</artifactId>
<version>5.10.0</version>
</dependency>
简单模型
在上图的模型中,有以下概念:
- 生产者,也就是要发送消息的程序
- 消费者:消息的接受者,会一直等待消息到来。
- 消息队列:图中红色部分。类似一个邮箱,可以缓存消息;生产者向其中投递消息,消费者从其中取出消息。
所有的中间件技术都是基于
tcp/ip
协议基础之上构建新型的协议规范,只不过rabbitmq
遵循的是amqp
实现步骤:
- 创建连接工程
- 创建连接 connection
- 通过连接获取通道 Channel
- 通过通道创建交换机,声明队列,绑定关系,路由key,发送消息,和接收消息
- 准备消息内容
- 发送消息给队列 queue
- 关闭连接
- 关闭通道
生产者
public class Producer {
public static void main(String[] args) {
//1. 创建连接工程
ConnectionFactory connectionFactory = new ConnectionFactory();
//这里要使用自己的IP地址
connectionFactory.setHost("192.168.57.129");
connectionFactory.setPort(5672);
connectionFactory.setUsername("admin");
connectionFactory.setPassword("admin");
connectionFactory.setVirtualHost("/");
Connection connection = null;
Channel channel = null;
try {
//2. 创建连接 connection
connection = connectionFactory.newConnection("生产者");
//3. 通过连接获取通道 Channel
channel = connection.createChannel();
//4. 通过通道创建交换机,声明队列,绑定关系,路由key,发送消息
String quequeName = "queuel";
/**
* @params1 队列的名称
* @params2 是否要持久化 durable-false
* @params3 排他性,是否是独占独立
* @params4 是否自动删除,随着最后一个消费者消息完毕以后是否把队列自动删除
* @params5 携带的附属参数
*/
channel.queueDeclare(quequeName,false,false,false,null);
//5. 准备消息内容
String message = "Hello,Consumer";
//6. 发送消息给队列 queue
channel.basicPublish("",quequeName,null,message.getBytes());
System.out.println("消息发送成功");
} catch (Exception e) {
e.printStackTrace();
}finally {
//7. 关闭连接
if (channel != null && channel.isOpen()){
try {
channel.close();
} catch (Exception e) {
e.printStackTrace();
}
}
//8. 关闭通道
if (connection != null && connection.isOpen()){
try {
connection.close();
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
}
消费者
public class Consumer {
public static void main(String[] args) {
//1. 创建连接工程
ConnectionFactory connectionFactory = new ConnectionFactory();
//这里要使用自己的IP地址
connectionFactory.setHost("192.168.57.129");
connectionFactory.setPort(5672);
connectionFactory.setUsername("admin");
connectionFactory.setPassword("admin");
connectionFactory.setVirtualHost("/");
Connection connection = null;
Channel channel = null;
try {
//2. 创建连接 connection
connection = connectionFactory.newConnection("消费者");
//3. 通过连接获取通道 Channel
channel = connection.createChannel();
//4. 通过通道创建交换机,声明队列,绑定关系,路由key,发送消息,和接收消息
String quequeName = "queue1";
channel.queueDeclare(quequeName,false,false,false,null);
//5.监听消息
DefaultConsumer consumer = new DefaultConsumer(channel){
/*
consumerTag:消息者标签,channel.basicConsume可以指定
envelope:消息包内容,可从中获取消息id,消息routing key,交换机,消息和重装标记(收到消息失败后是否需要重新发送)
properties:消息属性
body;消息
*/
@Override
public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
//路由key
System.out.println("路由key为:"+ envelope.getRoutingKey());
//交换机
System.out.println("交换机为:"+ envelope.getExchange());
//消息id
System.out.println("消息id为:"+ envelope.getDeliveryTag());
//收到的消息
System.out.println("接收到的消息:"+ new String(body,"UTF-8"));
System.out.println("");
System.out.println("======================================================");
System.out.println("");
}
};
channel.basicConsume("queue1", true, consumer);
} catch (Exception e) {
e.printStackTrace();
}finally {
//6. 不关闭资源,一直监听
}
}
}
AMQP
概念介绍
AMQP 一个提供统一消息服务的应用层标准高级消息队列协议,是应用层协议的一个开放标准,为面向消息的中间件设计。
AMQP是一个二进制协议,拥有一些现代化特点:
多信道
、协商式
,异步
,安全
,扩平台
,中立
,高效
。RabbitMQ 是 AMQP协议 的 Erlang的实现。
概念 | 说明 |
---|---|
连接 Connection | 一个网络连接,例如:TCP/IP套接字连接。 |
会话 Session | 端点之间的命名对话。在一个会话上下文中,保证“恰好传递一次”。 |
信道 Channel | 多路复用连接中的一条独立的双向数据流通道。为会话提供物理传输介质。 |
客户端 Client | AMQP连接或者会话的发起者。AMQP是非对称的,客户端生产和消费消息,服务器存储和路由这些消息。 |
服务节点Broker | 消息中间件的服务节点。一般情况下可以将一个RabbitMQ Broker看作一台RabbitMQ 服务器。 |
端点 | AMQP对话的任意一方。一个AMQP连接包括两个端点(一个是客户端,一个是服务器)。 |
消费者 Consumer | 一个从消息队列里请求消息的客户端程序。 |
生产者 Producer | 一个向交换机发布消息的客户端应用程序。 |
RabbitMQ运转流程
以 入门案例 为例
生产者发送消息
- 生产者创建连接(Connection),开启一个信道(Channel),连接到RabbitMQ Broker;
- 声明队列、设置属性;如是否排它,是否持久化,是否自动删除;
- 将路由键(空字符串)与队列绑定起来;
- 发送消息至RabbitMQ Broker;
- 关闭信道;
- 关闭连接;
消费者接收消息
- 消费者创建连接(Connection),开启一个信道(Channel),连接到RabbitMQ Broker
- 向Broker 请求消费相应队列中的消息,设置相应的回调函数;
- 等待Broker回应闭关投递响应队列中的消息,消费者接收消息;
- 确认(ack,自动确认)接收到的消息;
- RabbitMQ从队列中删除相应已经被确认的消息;
- 关闭信道;
- 关闭连接;
生产者流转过程解析
- 客户端与代理服务器Broker建立连接。调用
newConnection()
方法 , 会进一步封装Protocol Header 0-9-1
的报文头发送给Broker
,以此通知Broker
本次交互采用的是AMQP 0-9-1
协议,紧接着Broker
返回Connection.Start
来建立连接,在连接的过程中涉及Connection.Start/.Start-OK
、Connection.Tune/.Tune-Ok
,Connection.Open/ .Open-Ok
这6 个命令的交互。 - 客户端调用
connection.createChannel
方法。此方法开启信道,其包装的channel.open
命令发送给Broker
, 等待channel.basicPublish
方法,对应的AMQP命令为Basic.Publish
, 这个命令包含了content Header
和content Body()
。content Header 包含了消息体的属性,例如:投递模式,优先级等,content Body 包含了消息体本身。 - 客户端发送完消息需要关闭资源时,涉及到
Channel.Close和Channl.Close-Ok
与Connetion.Close和Connection.Close-Ok
的命令交互。
消费者流转过程解析
- 消费者客户端与代理服务器Broker建立连接。会调用
newConnection()
方法,这个方法会进一步封装Protocol Header 0-9-1
的报文头发送给Broker ,以此通知Broker 本次交互采用的是AMQP 0-9-1
协议,紧接着Broker 返回Connection.Start
来建立连接,在连接的过程中涉及Connection.Start/.Start-OK 、Connection.Tune/.Tune-Ok ,Connection.Open/ .Open-Ok
这6 个命令的交互。 - 消费者客户端调用
connection.createChannel
方法。和生产者客户端一样,协议涉及Channel . Open/Open-Ok
命令。 - 在真正消费之前,消费者客户端需要向Broker 发送
Basic.Consume
命令(即调用channel.basicConsume
方法〉将Channel 置为接收模式,之后Broker 回执Basic . Consume - Ok
以告诉消费者客户端准备好消费消息。 - Broker 向消费者客户端推送(Push) 消息,即
Basic.Deliver
命令,这个命令和Basic.Publish
命令一样会携带Content Header 和Content Body。
- 消费者接收到消息并正确消费之后,向Broker 发送确认,即
Basic.Ack
命令。 - 客户端发送完消息需要关闭资源时,涉及到
Channel.Close和Channl.Close-Ok 与Connetion.Close和Connection.Close-Ok
的命令交互。
个人博客为:
MoYu's HomePage
RabbitMQ入门案例的更多相关文章
- 2.RABBITMQ 入门 - WINDOWS - 生产和消费消息 一个完整案例
关于安装和配置,见上一篇 1.RABBITMQ 入门 - WINDOWS - 获取,安装,配置 公司有需求,要求使用winform开发这个东西(消息中间件),另外还要求开发一个日志中间件,但是也是要求 ...
- RabbitMQ入门-从HelloWorld开始
从读者的反馈谈RabbitMQ 昨天发完<RabbitMQ入门-初识RabbitMQ>,我陆陆续续收到一些反馈.鉴于部分读者希望结合实例来讲 期待下篇详细,最好结合案例.谢谢! 哪都好,唯 ...
- RabbitMQ入门:发布/订阅(Publish/Subscribe)
在前面的两篇博客中 RabbitMQ入门:Hello RabbitMQ 代码实例 RabbitMQ入门:工作队列(Work Queue) 遇到的实例都是一个消息只发送给一个消费者(工作者),他们的消息 ...
- RabbitMQ入门教程(十七):消息队列的应用场景和常见的消息队列之间的比较
原文:RabbitMQ入门教程(十七):消息队列的应用场景和常见的消息队列之间的比较 分享一个朋友的人工智能教程.比较通俗易懂,风趣幽默,感兴趣的朋友可以去看看. 这是网上的一篇教程写的很好,不知原作 ...
- SpringMVC入门案例及请求流程图(关于处理器或视图解析器或处理器映射器等的初步配置)
SpringMVC简介:SpringMVC也叫Spring Web mvc,属于表现层的框架.Spring MVC是Spring框架的一部分,是在Spring3.0后发布的 Spring结构图 Spr ...
- SpringMvc核心流程以及入门案例的搭建
1.什么是SpringMvc Spring MVC属于SpringFrameWork的后续产品,已经融合在Spring Web Flow里面.Spring 框架提供了构建 Web 应用程序的全功能 M ...
- Struts2第一个入门案例
一.如何获取Struts2,以及Struts2资源包的目录结构的了解 Struts的官方地址为http://struts.apache.org 在他的主页当中,我们可以通过左侧的Apache ...
- MyBatis入门案例、增删改查
一.MyBatis入门案例: ①:引入jar包 ②:创建实体类 Dept,并进行封装 ③ 在Src下创建大配置mybatis-config.xml <?xml version="1.0 ...
- Hibernate入门案例及增删改查
一.Hibernate入门案例剖析: ①创建实体类Student 并重写toString方法 public class Student { private Integer sid; private I ...
随机推荐
- 【luogu P3807】【模板】卢卡斯定理/Lucas 定理(含 Lucas 定理证明)
[模板]卢卡斯定理/Lucas 定理 题目链接:luogu P3807 题目大意 求 C(n,n+m)%p 的值. p 保证是质数. 思路 Lucas 定理内容 对于非负整数 \(n\),\(m\), ...
- 企业该选择什么样的CRM系统
不论您是需要CRM系统来优化业务流程,还是准备更换一款新的CRM系统,在这之前都应该先明确企业的需求,并了解CRM的哪些功能能够对企业有所帮助.例如,企业的管理者想了解每个销售人员的业绩情况,那么就应 ...
- MySql:Windows10安装mysql-8.0.18-winx64步骤
步骤: 1. 首先在安装的mysql目录下创建my.ini文件 (深坑)注意:my.ini必须保存为ANSI格式!!! 可以先创建一个my.txt的文件,然后另存为ANSI格式的文件! my.ini内 ...
- springboot项目启动,停止,重启
参考博客 https://www.cnblogs.com/c-h-y/p/10460061.html 打包插件,可以指定启动类 <build> <plugins> <pl ...
- Louvain 论文笔记
Louvain Introduce Louvain算法是社区发现领域中经典的基于模块度最优化的方法,且是目前市场上最常用的社区发现算法.社区发现旨在发现图结构中存在的类簇(而非传统的向量空间). Al ...
- Java并发实战一:线程与线程安全
从零开始创建一家公司 Java并发编程是Java的基础之一,为了能在实践中学习并发编程,我们跟着创建一家公司的旅途,一起来学习Java并发编程. 进程与线程 由于我们的目标是学习并发编程,所以我不会把 ...
- mongodb的基本命令与常规操作
1. 查看当前数据库的版本号:db.version()2. 查看当前所在数据库:db 默认是test数据库3. 查看当前数据库的连接地址:db.getMongo()4. 查看所有数据库:show da ...
- Mysql常用语句整理
把工作常用的mysql命令整理一下,省的用的时候在到处找 1.常用命令 1.1 登录 mysql -u root -p 1.2 生成随机数 若在 i<=R<=j 范围内生成随机数 FLOO ...
- CTF反序列化逃逸
刷了一下CTF反序列化的题,去年没有好好了解.又补了一次PHP,害太菜了.每天看看别人的博客真的可以鼓舞人.简单记录一下两道字符串逃逸问题 推荐一个反序列化总结的很好的笔记https://www.cn ...
- RSTP_PA协商过程
P/A协商的基本需求: P:①DP端口,②discarding A:①P2P链路 所有交换机的stp mode改为rstp,确保sw2的g0/0/3为AP,sw3的g0/0/3为DP 把sw3的g0/ ...