.NET6 RabbitMQ自动重连

序言

        在技术领域,不断学习和探索是保持竞争力的关键。最近,我重新开始了对RabbitMQ的研究,这个过程让我又收获了许多新的知识和技能,我觉得有必要记录下来,以便将来回顾和分享。

问题引出

        RabbitMQ是一款开源的消息队列中间件,它提供了高效、可靠、灵活的通信机制,广泛应用于分布式系统中。然而,有时候在使用RabbitMQ时会遇到连接断开的问题,这可能会导致消息传递中断和应用程序的不可用性。

问题分析

        当使用RabbitMQ时,可能会遇到以下几种情况导致连接断开的问题:

1.网络问题:网络中断、防火墙设置等问题可能导致RabbitMQ连接断开。

2.长时间空闲:如果连接在一段时间内没有进行任何通信,RabbitMQ可能会自动关闭连接。

3.RabbitMQ服务器问题:RabbitMQ服务器可能会因为负载过高或其他原因主动关闭连接。

解决方案

        虽然RabbitMQ.Client库,有心跳机制,有断线重连机制,但是在网络断开的时候,并不能重连。

        下面这段代码经过本人验证有效,可以解决上面的问题。

using RabbitMQ.Client.Events;
using RabbitMQ.Client;
using System.Text;
using RabbitMQDemo.Shared;
using System.Collections.Concurrent;
using RabbitMQ.Client.Exceptions;

namespace RabbitMQConsumerDemo
{
    public class RabbitMQRpcClientHandler
    {
        /// <summary>
        /// 定义一个静态变量来保存类的实列
        /// </summary>
        private static RabbitMQRpcClientHandler? _uniqueInstance;

        /// <summary>
        /// 定义一个标识确保线程同步
        /// </summary>
        private static readonly object _locker = new();

        /// <summary>
        /// Main entry point to the RabbitMQ .NET AMQP client API. Constructs RabbitMQ.Client.IConnection instances.
        /// </summary>
        private static IConnectionFactory? _factory;

        /// <summary>
        /// Main interface to an AMQP connection.
        /// </summary>
        private IConnection? _connection;

        /// <summary>
        /// 发送通道(发送通道及接收通道分开,避免互相影响,导致整个服务不可用)
        /// </summary>
        private IModel? _sendChannel;

        /// <summary>
        /// 接收通道(发送通道及接收通道分开,避免互相影响,导致整个服务不可用)
        /// </summary>
        private IModel? _listenChannel;

        /// <summary>
        /// 监听消费者
        /// </summary>
        private EventingBasicConsumer? _listenConsumer;

        /// <summary>
        /// 响应消费者
        /// </summary>
        private EventingBasicConsumer? _replyConsumer;

        /// <summary>
        /// RabbitMQ 主机域名
        /// </summary>
        private readonly string _defaultRabbitMQHostName = "127.0.0.1";

        /// <summary>
        /// RabbitMQ 服务器端口, 默认 5672, 网页监控页面是 15672
        /// </summary>
        private readonly int _defaultRabbitMQPort = 5672;

        /// <summary>
        /// RabbitMQ 用户名, 默认 guest
        /// </summary>
        private readonly string _defaultRabbitMQUserName = "guest";

        /// <summary>
        /// RabbitMQ 密码, 默认 guest
        /// </summary>
        private readonly string _defaultRabbitMQPassword = "guest!";

        /// <summary>
        /// 虚拟主机
        /// </summary>
        private readonly string _defaultRabbitMQVirtualHost = "/";

        /// <summary>
        /// 交换机
        /// </summary>
        private readonly string _exchangeName = "";

        /// <summary>
        /// 数据监控队列
        /// </summary>
        private readonly string _listenQueueName = "queue.listen.test";

        /// <summary>
        /// 指令响应队列
        /// </summary>
        private string _replyQueueName = string.Empty;

        /// <summary>
        /// 注册-路由键
        /// </summary>
        private readonly string _routingKeyRegister = "queue.register";

        /// <summary>
        /// 心跳-路由键
        /// </summary>
        private readonly string _routingKeyHeart = "queue.heart";

        /// <summary>
        /// 取消信号
        /// </summary>
        private readonly CancellationTokenSource _cts = new();

        /// <summary>
        /// 回调函数映射器
        /// </summary>
        private readonly ConcurrentDictionary<string, TaskCompletionSource<string>> _callbackMapper = new();

        private bool _connectionState;
        private bool _sendChannelState;
        private bool _listenChannelState;

        /// <summary>
        /// 连接状态
        /// </summary>
        public bool ConnectionState
        {
            get { return _connectionState; }
            set 
            {
                if (_connectionState == value)
                {
                    return;
                }

                _connectionState = value;
                if (_connectionState)
                {
                    Console.WriteLine($"{DateTime.Now:HH:mm:ss.fff}: 连接已打开");
                }
                else
                {
                    Console.WriteLine($"{DateTime.Now:HH:mm:ss.fff}: 连接已关闭");
                }
            }
        }

        /// <summary>
        /// 发送通道状态
        /// </summary>
        public bool SendChannelState
        {
            get { return _sendChannelState; }
            set 
            {
                if (_sendChannelState == value)
                {
                    return;
                }

                _sendChannelState = value;
                if (_sendChannelState)
                {
                    Console.WriteLine($"{DateTime.Now:HH:mm:ss.fff}: 发送通道已打开");
                }
                else
                {
                    Console.WriteLine($"{DateTime.Now:HH:mm:ss.fff}: 发送通道已关闭");
                }
            }
        }

        /// <summary>
        /// 接收通道状态
        /// </summary>
        public bool ListenChannelState
        {
            get { return _listenChannelState; }
            set 
            {
                if (_listenChannelState == value)
                {
                    return;
                }

                _listenChannelState = value;
                if (_listenChannelState)
                {
                    Console.WriteLine($"{DateTime.Now:HH:mm:ss.fff}: 接收通道已打开");
                }
                else
                {
                    Console.WriteLine($"{DateTime.Now:HH:mm:ss.fff}: 接收通道已关闭");
                }
            }
        }

        /// <summary>
        /// 定义私有构造函数,使外界不能创建该类实例 
        /// </summary>
        private RabbitMQRpcClientHandler()
        {
            // 创建连接工厂
            _factory = new ConnectionFactory()
            {
                HostName = _defaultRabbitMQHostName,//MQ服务器地址
                Port = _defaultRabbitMQPort,//MQ服务端口号
                UserName = _defaultRabbitMQUserName,//账号
                Password = _defaultRabbitMQPassword,//密码
                VirtualHost = _defaultRabbitMQVirtualHost,
                RequestedHeartbeat = TimeSpan.FromSeconds(2),
                AutomaticRecoveryEnabled = true,//自动重连
                To
<think>好的,用户问的是如何在.NET Framework中使用RabbitMQ。首先,我需要确认用户对RabbitMQ的基本了解程度。可能用户已经知道RabbitMQ是一个消息队列系统,但需要具体的实现步骤。接下来,我需要考虑.NET Framework的版本,因为不同版本可能有不同的库支持。 首先,应该从RabbitMQ的基础开始,比如安装和运行RabbitMQ服务器。用户可能需要先安装Erlang,然后安装RabbitMQ,并确保服务启动。这部分需要简明扼要的步骤说明,避免用户在这里卡住。 然后是.NET Framework项目的配置。用户需要安装RabbitMQ客户端库,这里应该推荐使用NuGet包管理器安装RabbitMQ.Client,因为这是最常用的方式。需要注意不同版本的兼容性,特别是.NET Framework 4.5.1及以上,可能需要确认版本号。 接下来是代码示例部分,分为生产者和消费者。需要提供基本的连接工厂设置,包括主机名、用户名、密码等。在发送消息时,要演示如何声明队列、将消息转换为字节流并发送。对于消费者,可能需要说明BasicConsume的使用方法,以及如何处理接收到的消息。 还需要提到一些注意事项,比如确保服务运行、异常处理、资源释放,以及消息确认机制。用户可能在实际应用中遇到连接问题或者消息未正确消费的情况,所以强调这些点很要。 另外,用户可能有扩展需求,比如持久化、交换机使用、集群配置等,但作为基础介绍,这部分可以简要提及,让用户知道有更多高级功能可以探索。 最后,检查是否所有步骤都清晰,没有遗漏关键点,比如NuGet安装、必要的using语句、连接参数的配置等。确保代码示例正确,没有语法错误,并且符合中文注释的要求。同时,提醒用户在生产环境中需要注意安全性,比如密码管理和异常处理。</think>在.NET Framework中使用RabbitMQ需要以下步骤: ### 一、环境准备 1. 安装RabbitMQ服务器 - 下载Erlang:https://www.erlang.org/downloads - 下载RabbitMQ:https://www.rabbitmq.com/download.html - 启用管理插件(命令行执行): $$ rabbitmq-plugins enable rabbitmq_management $$ 2. 创建.NET项目 - 新建控制台应用程序(.NET Framework 4.5.1+) - 通过NuGet安装客户端库: $$ Install-Package RabbitMQ.Client $$ ### 二、生产者示例代码 ```csharp using RabbitMQ.Client; // 创建连接工厂 var factory = new ConnectionFactory() { HostName = "localhost", UserName = "guest", // 默认用户名 Password = "guest" // 默认密码 }; // 建立连接 using(var connection = factory.CreateConnection()) using(var channel = connection.CreateModel()) { // 声明队列 channel.QueueDeclare(queue: "hello", durable: false, exclusive: false, autoDelete: false, arguments: null); // 发送消息 string message = "Hello RabbitMQ!"; var body = Encoding.UTF8.GetBytes(message); channel.BasicPublish(exchange: "", routingKey: "hello", basicProperties: null, body: body); Console.WriteLine($"已发送:{message}"); } ``` ### 三、消费者示例代码 ```csharp using RabbitMQ.Client; using RabbitMQ.Client.Events; var factory = new ConnectionFactory() { HostName = "localhost" }; using(var connection = factory.CreateConnection()) using(var channel = connection.CreateModel()) { channel.QueueDeclare(queue: "hello", durable: false, exclusive: false, autoDelete: false, arguments: null); var consumer = new EventingBasicConsumer(channel); consumer.Received += (model, ea) => { var body = ea.Body.ToArray(); var message = Encoding.UTF8.GetString(body); Console.WriteLine($"收到消息:{message}"); }; channel.BasicConsume(queue: "hello", autoAck: true, consumer: consumer); Console.WriteLine("等待消息..."); Console.ReadLine(); } ``` ### 四、关键概念说明 1. **连接管理** - 使用`ConnectionFactory`建立连接参数 - 建议使用`using`语句确保资源释放 2. **消息确认机制** - 自动确认(autoAck: true) - 手动确认(需调用`channel.BasicAck`) 3. **队列参数** ```csharp channel.QueueDeclare( queue: "队列名称", // 必填 durable: true, // 是否持久化 exclusive: false, // 是否排他队列 autoDelete: false, // 无消费者时自动删除 arguments: null); // 扩展参数 ``` ### 五、常见问题处理 1. **连接失败** - 检查RabbitMQ服务是否运行 - 确认防火墙开放5672端口 2. **消息堆积** - 增加消费者数量 - 使用QoS限制预取数量: $$ channel.BasicQos(0, 1, false); $$ 3. **消息持久化** ```csharp var properties = channel.CreateBasicProperties(); properties.Persistent = true; // 消息持久化 ``` ### 六、扩展应用 1. 使用交换机(Exchange)实现发布/订阅模式 2. 配置消息TTL(生存时间) 3. 实现死信队列(Dead Letter Exchange) 4. 集群部署方案 建议通过RabbitMQ管理界面(http://localhost:15672)监控消息状态,默认账号:guest/guest。生产环境务必修改默认凭证,建议使用SSL加密连接。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值