RabbitMQ集群搭建详细介绍以及解决搭建过程中的各种问题——实操型

RabbitMQ集群搭建详细介绍以及解决搭建过程中的各种问题——实操型

  • 1. 准备工作
    • 1.1 安装RabbitMQ
    • 1.2 简单部署搭建设计
    • 1.3 参考官网
  • 2. RabbitMQ 形成集群的方法
  • 3. 搭建RabbitMQ集群
    • 3.1 部署架构
    • 3.2 rabbitmq集群基础知识
      • 3.2.1 关于节点名称(标识符)
        • 3.2.1.1 节点名称
        • 3.2.1.2 主机解析
      • 3.2.2 端口访问
      • 3.2.3 群集中的节点
        • 3.2.3.1 配置 Erlang Cookie
    • 3.3 搭建集群(步骤)
      • 3.3.1 开放4369端口、25672端口
      • 3.3.2 修改每台机器的 /etc/hosts 文件
      • 3.3.3 停止服务、配置 Erlang Cookie
      • 3.3.4 启动独立节点
      • 3.3.5 创建集群
      • 3.3.5.1 创建集群步骤
      • 3.3.5.2 遇到的问题
        • 3.3.5.2.1 问题1:
        • 3.3.5.2.1 问题2——移除节点之后重新加入的问题:
      • 3.3.6 查看集群状态(命令查看)
      • 3.3.7 查看集群状态(UI界面查看)
    • 3.4 为集群创建用户
    • 3.5 集群——简单实用
      • 3.5.1 查看是否启用了自动集群发现机制
      • 3.5.2 自动集群发现机制(简单演示)
    • 3.6 解除集群节点
  • 4. 镜像队列
    • 4.1 普通模式集群存在的问题
      • 4.1.1 演示问题——不能备份
      • 4.1.2 引入镜像队列

1. 准备工作

1.1 安装RabbitMQ

  • 准备3台机器,分别安装上RabbitMQ,这里就不多说了,关于安装可以参考下面的文章,里面写的很详细,如下:
    RabbitMQ入门(RabbitMQ的安装 + 解决RabbitMQ安装过程中的版本等问题 + 解决启动RabbitMQ服务失败等问题 + Java客户端使用).

1.2 简单部署搭建设计

  • 服务器情况

    节点(理解用)节点名称(计算生成)主机名主机IP
    node1rabbit@VM-24-15-centosVM-24-15-centos49.xxx.99
    node2rabbit@hello-TQ2hello-TQ215xxxx212
    node3rabbit@hello_TQ1hello_TQ143.xxx.116
  • 部署设计,如下:
    在这里插入图片描述

1.3 参考官网

  • 找官网参考,如下:
    https://www.rabbitmq.com/clustering.html#node-count.

2. RabbitMQ 形成集群的方法

  • 官方说了以下几种,感兴趣的可以试试
    在这里插入图片描述
  • 接下来介绍的是最好一种,手动使用 rabbitmqctl

3. 搭建RabbitMQ集群

3.1 部署架构

  • 这里详细见1.2中的介绍。

3.2 rabbitmq集群基础知识

3.2.1 关于节点名称(标识符)

3.2.1.1 节点名称
  • RabbitMQ 节点由节点名称标识。节点名称由两部分组成,前缀(通常是 rabbit)和主机名。例如,rabbit@node1.messaging.svc.local 是一个节点名称,前缀为 rabbit,主机名为 node1.messaging.svc.local。再例如:rabbit@VM-24-15-centosrabbit@hello_TQ1
  • 在群集中,节点使用节点名称相互标识和联系。这意味着必须解析每个节点名称的主机名部分。CLI 工具还使用节点名称识别和寻址节点。
  • 当节点启动时,它会检查是否已为其分配节点名称。这是通过RABBITMQ_NODENAME环境变量完成的。如果未显式配置任何值,则节点将解析其主机名,并将 rabbit 附加到它以计算其节点名称
3.2.1.2 主机解析
  • 如下:
    在这里插入图片描述

3.2.2 端口访问

  • 开放4369端口、25672端口,关于相关端口,如下:
    在这里插入图片描述

3.2.3 群集中的节点

3.2.3.1 配置 Erlang Cookie
  • CLI工具如何对节点进行身份验证(以及节点之间的身份验证)
    • RabbitMQ 节点和 CLI 工具(例如 rabbitmqctl)使用 cookie 来确定它们是否被允许相互通信。为了使两个节点能够通信,它们必须具有相同的共享密钥,称为Erlang cookie。Cookie 只是一个字母数字字符字符串,最大为 255 个字符。它通常存储在本地文件中。该文件必须仅供所有者访问(例如,具有 600 或类似的 UNIX 权限)。每个群集节点必须具有相同的 Cookie
  • 关于cookie 文件的位置:
    • RabbitMQ 服务启动时,erlang VM 会自动创建该 cookie 文件,cookie通常位于 /var/lib/rabbitmq/.erlang.cookie(由服务器使用)和$HOME/.erlang.cookie(由CLI工具使用)中。
      在这里插入图片描述
  • 所以,下面搭建集群时,需要统一一下.erlang.cookie文件(要求相同的 Cookie)
    • 将node1机器上的.erlang.cookie文件分别拷贝到node2和node3上,拷贝时候记得停止RabbitMQ服务。

3.3 搭建集群(步骤)

3.3.1 开放4369端口、25672端口

  • 云服务器的话,开放 4369、25672 端口,如果防火墙是开启的话,执行下面命令:
    firewall-cmd --state  # 查看防火墙的状态sudo firewall-cmd --add-port=4369/tcp --permanent
    sudo firewall-cmd --add-port=25672/tcp --permanentfirewall-cmd --reloadfirewall-cmd --list-all
    
  • 如果不开放此端口,后面执行 rabbitmqctl stop_app 时报错,不能链接此端口。

3.3.2 修改每台机器的 /etc/hosts 文件

  • vim /etc/hosts,让集群节点间需能互相访问,故每个集群节点的hosts文件应包含集群内所有节点的信息以保证互相解析。
    • 需要注意的是:这里的主机名不是随便填写的,你要看你3台服务器的主机名是什么,比如,我在node1节点的主机名是VM-24-15-centos,node2节点的主机名是hello-TQ2,node3节点的主机名是hello_TQ1,所以我在每台服务器上的配置如下:
      服务器IP1 VM-24-15-centos
      服务器IP2  hello-TQ2
      服务器IP3 hello_TQ1
      
      在这里插入图片描述
  • 重启网络,便于系统识别hosts
    systemctl restart network
    
  • 备注:如果你是刚初始化的服务器,可以修改主机名,但是如果你服务器上已经装了很多东西,就没必要修改主机名了(谨慎修改),关于修改主机名的,可以看下面的文章:
    云服务器修改主机名hostname.

3.3.3 停止服务、配置 Erlang Cookie

  • 这个详细的话见上面说的,操作的话,只需将node1(我这里是VM-24-15-centos主机)上的.erlang.cookie文件分别拷贝到 node2 和 node3 对应的机器上。
  • 停止所有服务,构建Erlang的集群环境
    systemctl stop rabbitmq-server或 service rabbitmq-server stop
    
  • 开始拷贝.erlang.cookie
    scp /var/lib/rabbitmq/.erlang.cookie root@node2服务IP:/var/lib/rabbitmq/
    scp /var/lib/rabbitmq/.erlang.cookie root@node3服务IP:/var/lib/rabbitmq/
    

3.3.4 启动独立节点

  • 启动三台RabbitMQ,以正常方式在所有节点上启动 RabbitMQ,启动命令:
    rabbitmq-server -detached
    

3.3.5 创建集群

3.3.5.1 创建集群步骤

  • 为了将集群中的三个节点连接起来,我们告诉其中两个节点,比如rabbit@hello-TQ2rabbit@hello_TQ1,加入node1节点的集群,比如rabbit@VM-24-15-centos。在此之前,必须重置两个新加入的成员。
    • 分别在要加入的两个机上上,执行下面的操作命令:
      rabbitmqctl stop_app   # 1.关闭RabbitMQ服务rabbitmqctl reset   # 2.重置节点# 3.节点加入, 在一个node加入cluster之前,必须先停止该node的rabbitmq应用,即先执行stop_app
      rabbitmqctl join_cluster rabbit@VM-24-15-centosrabbitmqctl start_app    #  4.启动服务
      
    • 注意:
      • abbitmqctl stop 会将Erlang虚拟机关闭;
      • rabbitmqctl stop_app 只关闭rabbitMQ服务
      • rabbitmqctl start_app 只启动应用服务
  • 加入节点、重置节点注意:
    • 我们首先将 rabbit@hello-TQ2rabbit@hello_TQ1 加入一个集群。为此,我们要在 rabbit@hello-TQ2rabbit@hello_TQ1停止 RabbitMQ 应用程序并加入 rabbit@VM-24-15-centos(node1节点) 群集,然后重新启动 node2、node3的RabbitMQ 应用程序。
    • 请注意,节点必须重置后才能加入现有群集。重置节点会删除该节点上以前存在的所有资源和数据。这意味着,节点不能在成为群集成员的同时保留其现有数据。如果需要这样做,可以使用蓝/绿部署策略或备份和还原。
  • 操作如下:
    在这里插入图片描述
    在这里插入图片描述

3.3.5.2 遇到的问题

3.3.5.2.1 问题1:
  • 问题如下:
    unable to connect to epmd (port 4369) on VM-24-15-centos: nxdomain (non-existing domain)
    
    在这里插入图片描述
  • 解决问题,修改/etc/hosts,让集群节点间需能互相访问,故每个集群节点的hosts文件应包含集群内所有节点的信息以保证互相解析。如果按上面步骤操作应该不会报这个错,所以此处不做过多解释,详细见3.3.2节
3.3.5.2.1 问题2——移除节点之后重新加入的问题:
  • 问题描述:第一次加没问题,移除再次加入直接报错,在执行rabbitmqctl reset 命令时报错,错误如下:
    Error:
    {:cannot_start_mnesia, {{:shutdown, {:failed_to_start_child, :mnesia_kernel_sup, :killed}}, {:mnesia_app, :start, [:normal, []]}}}
    
    在这里插入图片描述
  • 解决问题:
    • 第一:先解决上面的问题,解决方法就是删除存储的数据,如下:
      cd /var/lib/rabbitmq/mnesia/rabbit@hello_TQ1
      rm -rf ./*cd /var/lib/rabbitmq
      rm -rf erl_crash.dump
      rm -rf MnesiaCore*
      
      在这里插入图片描述
    • 第二:解决之后,换了新错误,如下:
      	Error:
      {:corrupt_or_missing_cluster_files, {:error, :enoent}, {:error, :enoent}}
      
      • 解决问题,rest之前执行一下命令rabbitmqctl cluster_status,一连串的操作如下(上面那个应该也可以按这个解决,不用手动rm-rf,不知道没尝试,但是感觉可以):
        rabbitmq-server -detached
        rabbitmqctl start_app    # 先启
        rabbitmqctl cluster_status
        rabbitmqctl stop_app  # 再停
        rabbitmqctl reset   # 重置rabbitmqctl join_cluster rabbit@VM-24-15-centos
        rabbitmqctl start_app
        
        在这里插入图片描述

3.3.6 查看集群状态(命令查看)

  • 通过在任一节点上运行 rabbitmqctl cluster_status 命令,我们可以看到这两个节点已加入集群中:
    rabbitmqctl cluster_status
    
    在这里插入图片描述

3.3.7 查看集群状态(UI界面查看)

  • 如下:
    在这里插入图片描述

3.4 为集群创建用户

  • 如下,在任一台机器上创建即可,创建之后,3个IP都可以登录
    rabbitmqctl add_user mqAdmin2 123456rabbitmqctl set_user_tags mqAdmin2 administratorrabbitmqctl set_permissions -p "/" mqAdmin2 ".*" ".*" ".*"
    
    在这里插入图片描述
    在这里插入图片描述

3.5 集群——简单实用

3.5.1 查看是否启用了自动集群发现机制

  • rabbitmq集群发现机制:
    • 如果集群中默认配置中启用了RabbitMQ的自动集群发现机制。那么在一个节点上创建Exchange或Queue时,它会自动地在整个集群中被发现。
    • 当您在一个节点上创建Exchange或Queue时,该节点会将其信息广播到整个集群中,其他节点会收到这个广播消息并自动地创建相应的Exchange或Queue。这种方式可以使得集群中的Exchange和Queue自动地分布到所有节点上,从而提高系统的可用性和可伸缩性。
  • 另外前提:我这里的版本是 RabbitMQ 3.9.13 Erlang 23.3.4.18,没有找到默认的 rabbitmq.conf 文件 或 rabbitmq.conf.d文件。
    • 一般情况下,这两个文件在/etc/rabbitmq目录下,如果有这个配置文件的,可以查看配置文件里的 cluster_formation.peer_discovery_backend 行的配置,如果该行的值为rabbit_peer_discovery_classic_config,则表示启用了自动集群发现机制。如果该行的值为其他任何值,则表示未启用自动集群发现机制。
    • 如果没有配置文件的,也可通过 rabbitmqctl environment 命令查看配置,如下:
      rabbitmqctl environment | grep peer_discovery_backend
      
      在这里插入图片描述

3.5.2 自动集群发现机制(简单演示)

  • 如下:在node1上和node3上分别创建一个队列,其他节点上都能发现:
    在这里插入图片描述
  • 并且可以消费,消费一次,在node2上消费之后,其他节点就不会再有此已消费的消息。

3.6 解除集群节点

  • 如果想解除node3(rabbit@hello_TQ1):
    • 首先,在node3机器上做如下操作,命令如下:
      rabbitmqctl stop_app    # 关闭rabbitMQ服务
      
    • 然后,在node1上做忘记node3节点的操作,如下:
      rabbitmqctl forget_cluster_node rabbit@hello_TQ1
      
      操作此命令之前,一定要先关闭被忘记的节点的服务,否则,如下错误:
      在这里插入图片描述
  • 根据提示,可知,上面两步操作可以都在node1上完成,直接先stop,再forget,如下:
    rabbitmqctl -n rabbit@hello-TQ2 stop_apprabbitmqctl forget_cluster_node rabbit@hello-TQ2
    
    在这里插入图片描述

4. 镜像队列

4.1 普通模式集群存在的问题

4.1.1 演示问题——不能备份

  • 上面集群虽然可以启动自动集群发现机制,但是存在的问题是,如果创建队列(发送消息)的rabbitmq服务宕机了,消息就丢失了,比如上面node3节点上创建了队列发送了消息,其他node可以发现,但是如下现在node3宕机了,再看下效果:
    在这里插入图片描述
    在这里插入图片描述
  • 所以问题是:只能被发现,但是没有备份!

4.1.2 引入镜像队列

  • RabbitMQ的自动集群发现机制可以帮助集群中的Exchange和Queue自动地分布到所有节点上,从而提高系统的可用性和可伸缩性。然而,这种机制并不能完全保证消息的高可用性。如果某个节点出现故障,该节点上的Exchange和Queue可能会丢失或不可用,从而导致消息丢失或无法传递。为了解决这个问题,RabbitMQ提供了镜像队列(Mirrored Queue)的功能
  • 关于镜像队列的,见下篇!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/95688.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Java常见API---split()

package daysreplace;public class SplitTest {public static void main(String[] args) {String str"武汉市|孝感市|长沙市|北京市|上海市";String[] array str.split("\\|");System.out.println(array[0]);System.out.println(array[1]);System.out.pri…

【C++】一文带你走入vector

文章目录 一、vector的介绍二、vector的常用接口说明2.1 vector的使用2.2 vector iterator的使用2.3 vector空间增长问题2.4 vector 增删查改 三、总结 ヾ(๑╹◡╹)ノ" 人总要为过去的懒惰而付出代价ヾ(๑╹◡╹)ノ" 一、vector的介绍 vector…

golang gin——文件上传(单文件,多文件)

文件上传 单文件上传 从form-data获取文件 package uploadimport ("github.com/gin-gonic/gin""net/http" ) // 单文件上传,多文件上传 func Upload(c *gin.Context) {file, _ : c.FormFile("file") // file为字段名dst : "…

nodejs开发环境搭建

Nodejs是一个开源的、跨平台JavaScript运行时环境,其使用V8引擎对JavaScript脚本执行解释,在前后端分离的应用架构设计中,其既能支持web页面服务应用的开发、也能支持后端接口服务应用的开发,类似于Java语言的J2EE运行时环境&…

安装matplotlib__pygame,以pycharm调入模块

安装pip 安装matplotlib 安装完毕,终端输入pip list检查 导入模块出现bug,发现不是matplotlib包的问题,pycharm版本貌似不兼容,用python编辑器可正常绘图,pygame也可正常导入。 ​​​​​​​ pycharm版本问题解决 终…

【Spring笔记02】Spring中的IOC容器和DI依赖注入介绍

这篇文章,主要介绍一下Spring中的IOC容器和DI依赖注入两个概念。 目录 一、IOC控制反转 1.1、什么是IOC 1.2、两种IOC容器 (1)基于BeanFactory的IOC容器 (2)基于ApplicationContext的IOC容器 二、DI依赖注入 2.…

stm32-SPI协议

SPI协议详解(图文并茂超详细) SPI通讯协议 于是我们想有没有更好一点的串行通讯方式;相比较于UART,SPI的工作方式略有不同。 SPI是一个同步的数据总线,也就是说它是用单独的数据线和一个单独的时钟信号来保证发送端和…

MySQL中的 增 删 查 改(CRUD)

目录 新增 insert into 表名 value(数据,数据),.......; insert into 表名(列1,列2.....) value(数据,数据),.......; datatime 类型的数据如何插入? 查询 select * from 表名…

动态调整系统主题色(4): CssVar 与 Variant 方案的探索

动态调整系统主题色(4): CssVar 与 Variant 方案的探索 动态调整系统主题色(4): CssVar 与 Variant 方案的探索 前言方案的介绍与比较 CssVar (CSS 变量方案)CSS 变量方案与 tailwindcss 的结合Variant 方案 2种方案在小程序上的示例之前的几篇 前言 这篇已经是动态调整系统…

Docker 的数据管理与Docker 镜像的创建

------------------Docker 的数据管理--------------------- 管理 Docker 容器中数据主要有两种方式:数据卷(Data Volumes)和数据卷容器(DataVolumes Containers)。 1.数据卷 数据卷是一个供容器使用的特殊…

什么是Vue的JSX语法?如何使用JSX语法

Vue的JSX语法:更接近JavaScript的模板语言 Vue.js是一个流行的JavaScript框架,用于构建交互式的Web应用程序。虽然Vue通常使用模板语法来构建用户界面,但它也提供了JSX语法的支持,使开发人员能够更接近JavaScript的表达方式来构建…

分享几个优秀开源免费管理后台模版,建议收藏!

大家好,我是 jonssonyan 今天和大家分享一些免费开源的后台管理页面,帮助大家快速搭建前端页面。为什么要用模板?道理很简单,原因是方便我们快速开发。我们不应该花太多的时间在页面调整上,而应该把精力放在核心逻辑和…

re学习(37)DASCTF 2023 0X401七月暑期挑战赛 controflow

程序通过改变栈里面的返回地址来控制程序的控制流 从而达到混淆的效果 左侧有许多被hook的函数 在每个函数开头设置断点 然后观察程序的运行流程 会发现输入的数据会进行 异或 相加 异或 相减 相乘 异或等操作 要注意部分运算的索引是 从[10]开始的 具体思路参考&#xf…

黑豹程序员-架构师学习路线图-百科:JSON替代XML

文章目录 1、数据交换之王2、XML的起源3、JSON诞生4、什么是JSON 1、数据交换之王 最早多个软件之间使用txt进行信息交互,缺点:纯文本,无法了解其结构;之后使用信令,如:电话的信令(拨号、接听、…

三十、【进阶】B-Trees的演变过程

1、索引结构 (1)二叉树 (2)B-Tree树 B-Tree树最大度数为5,代表每一个节点最多存储4个key(每个节点最多存储4个数据),5个指针(可以指向5个子节点)。 2、演变过程(最大度数为5) &…

数据挖掘(3)特征化

从数据分析角度,DM分为两类,描述式数据挖掘,预测式数据挖掘。描述式数据挖掘是以简介概要的方式描述数据,并提供数据的一般性质。预测式数据挖掘分析数据建立模型并试图预测新数据集的行为。 DM的分类: 描述式DM&#…

记录本地部署Stable-diffusion所依赖的repositories和一些插件

今天按照其他文章的步骤拉取好了https://github.com/AUTOMATIC1111/stable-diffusion-webui后,点击webui-user.bat后发现,repositories和models还得慢慢拉取,好吧,GitHub Desktop,启动! BLIP: https://git…

使用sqlmap获取数据步骤

文章目录 1.使用sqlmap获取所有数据库2.使用sqlmap获取当前连接数据库3.使用sqlmap获取当前数据库下所有表名4.使用sqlmap获取当前数据库下某个表下所有列名5.使用sqlmap获取当前数据库下某个表下指定字段的数据6.测试当前用户是否是管理员7.使用burpsqlmap批量检测8.脱库命令9…

c++使用ifstream和ofstream报错:不允许使用不完整的类型

学习《C Primer》关于IO库的部分&#xff0c;输入284页的的代码&#xff0c;出现了报错&#xff1a; 不允许使用不完整的类型 原来的代码&#xff1a; #include <iostream> #include <vector> using namespace std;int main(int argc, char **argv) {ifstream in…

Next.js 入门笔记

前言 之前初步体验了 React 的魅力, 又看文档理解了一下 useState 和 useEffect, 目前初步理解的概念是: useState 用来声明在组件中使用并且需要修改的变量 useEffect 用来对 useState 声明的变量进行初始化赋值 可能理解的不太准确, 不过大概差不多是这么个意思. 但是再往后…