RabbitMQ 学习笔记
环境:
MacOS 10.14
Node.js 8.9.1
零、背景
目前有个上线应用会接受多个请求,且每个请求的处理时间可能很久,可能到数小时,所以就想采用异步机制,至于复杂的运算就用消息队列(MQ)去慢慢消化。
网上调研了一圈,遂采用RabbitMQ。
一、安装
1、安装
(1) MacOS
brew install rabbitmq
(2) CentOS (Linux)
https://tecadmin.net/install-rabbitmq-on-centos/
2、配置环境变量
export PATH=$PATH:/usr/local/opt/rabbitmq/sbin
3、使用
(1) 服务器端
rabbitmq-server
启动需要(默认)200 MB的磁盘空间,但可以通过配置文件里的
disk_free_limit修改。
(2) 客户端
以 Node.js 为例:
npm i amqplib
https://www.npmjs.com/package/amqplib
var amqp = require('amqplib/callback_api');
4、用 rabbitmq management 进行后台管理
(1) 开启服务
rabbitmq-plugins enable rabbitmq_management
此时/etc/rabbitmq下会多出enabled_plugins文件,内容为:
[rabbitmq_management].
此时 rabbitmq management 的地址为http://localhost:15672,默认用户密码为 guest/guest。
但此时外网访问,登录时会提示 User can only log in via localhost 。
这是由于 rabbitmq 从 3.3.0 开始禁止使用 guest/guest 权限通过除 localhost 外的访问。
(2) 开启外网访问
1、rabbitmq 初始并没有创建配置文件,需要自行拷贝。
cp /usr/share/doc/rabbitmq-server-3.7.9/rabbitmq.config.example /etc/rabbitmq/rabbitmq.config
2、修改此配置文件rabbitmq.config
vim /etc/rabbitmq/rabbitmq.config
把 loopback_users 的注释解开:
%%{loopback_users, []} -> {loopback_users, []}
这里请小心 {loopback_users, []} 后的逗号可能需要去掉,不然格式会报错。
3、重启服务
sudo systemctl restart rabbitmq-server
注:可能在访问的时候会报这样的错:
解决办法: 关闭全局 ss 代理
二、使用
1、connection —— 连接
amqp.connect('amqp://localhost', function(error0, connection) {
if (error0) {
throw error0;
}
// ……
// connection.close();
});
2、channel —— 通道
通道分为:
生产者(发送者)
消费者(接收者)
connection.createChannel(function(error1, channel) {
if (error1) {
throw error1;
}
// channel ……
});
(1) queue —— 队列
队列里面塞入的是消息。
生产者
var queue = 'queue_name';
# 创建or连上队列
channel.assertQueue(queue, {
durable: true # 队列持久化
});
# 临时队列(当前 connection 断掉后就会被删除)—— 队列名随机
channel.assertQueue('',{ exclusive:true });
# 将消息塞入队列
channel.sendToQueue(queue, Buffer.from(msg), {
persistent: true # 消息持久化
});
关于持久化:
一个是防止服务器端的队列丢失,一个是防止服务器端的队列里的消息丢失。
但是这并不能避免:如果服务器端在RabbitMQ接受消息的过程中挂了导致的消息丢失。如果需要更强的保证,可以使用 发布者确认。
消费者
var queue = 'queue_name';
# 从队列取出消息
channel.consume(queue, function(msg) {
channel.ack(msg); # 发送确认信号
}, {
noAck: false
});
关于 ACK ( Acknowledgement )
noAck: true 则服务器端不会期望收到 ACK,也就是说,消息在被发送后会立即出列。
而 noAck: false 则需要消费者发送 ACK,即channel.ack(msg); ,但如果超时未回复 ACK,消息会重新排队(但如果同时有其他可用消费者,则会迅速安排过去)
查看当前有多少队列及各中有多少消息:
sudo rabbitmqctl list_queues
(2) prefetch —— 预取
channel.prefetch(1);
# 表示这个通道如果有{1}个未完成的消息,则不会接受新的消息
(3) exchange —— 交换
如果有多个队列,生产者的消息应该如何分配呢?这个时候就需要一个中间件——交换
其中交换类型有四种:“”(默认交换), topic, headers, fanout
A、 “”(默认交换)
RabbitMQ中消息传递模型的核心思想是生产者永远不会将任何消息直接发送到队列。所以不建议使用channel.sendToQueue(),此为 “”(默认交换)。
如果没有队列绑定到交换,消息将会丢失。
B、fanout(广播)
生产者
var exchange = 'logs';
# 创建or连上交换
channel.assertExchange(exchange, 'fanout', {
durable: false # 持久化
});
### 推消息给交换
channel.publish(exchange, '', Buffer.from(msg));
消费者
var exchange = 'logs';
# 创建or连上交换
channel.assertExchange(exchange, 'fanout', {
durable: false # 持久化
});
# ------------------------
# 绑定 交换+队列
channel.bindQueue('queue_1', exchange, '');
channel.bindQueue('queue_2', exchange, '');
channel.bindQueue('queue_3', exchange, '');
queue_1、queue_2、queue_3 都会收到相同的一条消息。
C、direct (直接)
生产者
var exchange = 'logs';
# 创建or连上交换
channel.assertExchange(exchange, 'fanout', {
durable: false # 持久化
});
### 推消息给交换
channel.publish(exchange, 'black', Buffer.from(msg));
消费者
var exchange = 'logs';
# 创建or连上交换
channel.assertExchange(exchange, 'fanout', {
durable: false # 持久化
});
# ------------------------
# 绑定 交换+队列
channel.bindQueue('queue_1', exchange, 'white');
channel.bindQueue('queue_2', exchange, 'black');
channel.bindQueue('queue_3', exchange, 'red');
只有 queue_2 才会收到消息。
D、topic
生产者
var exchange = 'logs';
# 创建or连上交换
channel.assertExchange(exchange, 'fanout', {
durable: false # 持久化
});
### 推消息给交换
channel.publish(exchange, 'kern.critical', Buffer.from(msg));
消费者
var exchange = 'logs';
# 创建or连上交换
channel.assertExchange(exchange, 'fanout', {
durable: false # 持久化
});
# ------------------------
# 绑定 交换+队列
channel.bindQueue('queue_1', exchange, '#');
channel.bindQueue('queue_2', exchange, "kern.*");
channel.bindQueue('queue_3', exchange, "*.critical");
- *(星号)可以替代一个单词。
- #(hash)可以替换零个或多个单词。
查看所有的 交换 及 交换绑定队列
sudo rabbitmqctl list_exchanges
sudo rabbitmqctl list_bindings
代码职责风格:
生产者只管发送消息就好 (比如发送消息给队列或者交换)
消费者要负责接受消息以外的更多事 (比如负责队列的 prefetch 设置,或者交换的绑定)
3、远程过程调用(RPC)
略
三、应用
例如可以用到日志系统中:对所有等级的日志都打印到控制台(即下面的队列),而 error 日志单独持久化到 disk(即上面的队列)。

四、MQ 的优缺点
优点:解耦、异步、削峰
缺点:系统可用性降低,系统复杂性增加
参考资料
1、官方RabbitMQ教程
https://www.rabbitmq.com/getstarted.html
2、amqp.node 参考API
https://www.squaremobius.net/amqp.node/channel_api.html#channel_ack
RabbitMQ 学习笔记的更多相关文章
- RabbitMQ学习笔记(五) Topic
更多的问题 Direct Exchange帮助我们解决了分类发布与订阅消息的问题,但是Direct Exchange的问题是,它所使用的routingKey是一个简单字符串,这决定了它只能按照一个条件 ...
- RabbitMQ学习笔记1-hello world
安装过程略过,一搜一大把. rabbitmq管理控制台:http://localhost:15672/ 默认账户:guest/guest RabbitMQ默认监听端口:5672 JAVA API地 ...
- (转) Rabbitmq学习笔记
详见原文: http://blog.csdn.net/shatty/article/details/9529463 Rabbitmq学习笔记
- 官网英文版学习——RabbitMQ学习笔记(十)RabbitMQ集群
在第二节我们进行了RabbitMQ的安装,现在我们就RabbitMQ进行集群的搭建进行学习,参考官网地址是:http://www.rabbitmq.com/clustering.html 首先我们来看 ...
- 官网英文版学习——RabbitMQ学习笔记(一)认识RabbitMQ
鉴于目前中文的RabbitMQ教程很缺,本博主虽然买了一本rabbitMQ的书,遗憾的是该书的代码用的不是java语言,看起来也有些不爽,且网友们不同人学习所写不同,本博主看的有些地方不太理想,为此本 ...
- RabbitMQ学习笔记五:RabbitMQ之优先级消息队列
RabbitMQ优先级队列注意点: 1.只有当消费者不足,不能及时进行消费的情况下,优先级队列才会生效 2.RabbitMQ3.5以后才支持优先级队列 代码在博客:RabbitMQ学习笔记三:Java ...
- RabbitMQ学习笔记(六) RPC
什么RPC? 这一段是从度娘摘抄的. RPC(Remote Procedure Call Protocol)——远程过程调用协议,它是一种通过网络从远程计算机程序上请求服务,而不需要了解底层网络技术的 ...
- 官网英文版学习——RabbitMQ学习笔记(八)Remote procedure call (RPC)
在第四篇学习笔记中,我们学习了如何使用工作队列在多个工作者之间分配耗时的任务. 但是,如果我们需要在远程计算机上运行一个函数并等待结果呢?这是另一回事.这种模式通常称为远程过程调用或RPC. ...
- 官网英文版学习——RabbitMQ学习笔记(二)RabbitMQ安装
一.安装RabbitMQ的依赖Erlang 要进行RabbitMQ学习,首先需要进行RabbitMQ服务的安装,安装我们可以根据官网指导进行http://www.rabbitmq.com/downlo ...
- RabbitMQ学习笔记一
前 言 -解决问题 一.RabbitMQ安装 1.安装erlang 环境 a.下载erlang 版本,注意这里需要和安装的rabbitMq版本相配对,rabbitMQ官方网站上可以查到:https ...
随机推荐
- python批量重命名【截取文件名前六个字符 】
#!/usr/bin/python # -*- coding: UTF-8 -*- import os, sys # 打开文件 path = "/home/landv/Desktop/l/& ...
- 关于tomcat WEB-INF/lib下类加载顺序
关于tomcat WEB-INF/lib下类加载顺序问题 问题描述 smc应用最近碰到一个线上问题,预发机器规则测试接口没问题,但是线上机器就是调用有问题,表面上看,maven模块引用的是自己想要的j ...
- RabbitMQ原理图
一.RabbitMQ 原理图 二.Rabbit 交换器讲解 1 Direct 交换器(发布与订阅 完全匹配) 1.2搭建环境 1 ...
- pytest简介
pytest有哪些优点? 允许直接使用assert进行断言,而不需要使用self.assert*; 可以自动寻找单测文件.类和函数; Modular fixtures可以用于管理小型或参数化的测试信息 ...
- myBase Desktop 6.5.1 无限期试用
清空安装目录下的"nyfedit.ini"文件的"App.UserLic.FirstUseOn="配置项的值
- idea通过mapper快速定位到xml文件
1.点击File找到设置(Settings) 2.点击Plugins下的 Browse respositories 3.在搜索栏搜索mybatis ,选中 Free Mybatis plugin——i ...
- Shell脚本学习 - 流程控制和函数
继续Shell的学习.上两篇是关于基本数据类型,基本语法以及运算符相关,这一篇是流程控制相关(if, for, while) 流程控制 if else 流程控制不可为空,如果else没有语句执行,就不 ...
- 封装PDO函数
funPDO.php <?php /** * @title: 封装PDO函数 * * @Features: * 1. 封装 SELECT ,INSERT,DELETE,UPDATE 操作 @do ...
- String StringBuffer StringBulider 详细看https://www.cnblogs.com/su-feng
主要区别:运行速度和线程安全 StringBuilder > StringBuffer > String String最慢是因为字符串常量不可改变,例如 str +“cccc” 如果 ...
- spring为什么推荐使用构造器注入?
闲谈 Spring框架对Java开发的重要性不言而喻,其核心特性就是IOC(Inversion of Control, 控制反转)和AOP,平时使用最多的就是其中的IOC,我们通过将组件交由Spr ...
