C#隊列學習筆記:RabbitMQ優先順序隊列

来源:https://www.cnblogs.com/atomy/archive/2020/04/11/12672396.html
-Advertisement-
Play Games

一、引言 在具體業務中可能會遇到一些要提前處理的消息,比如普通客戶的消息按先進先出的順序處理,Vip客戶的消息要提前處理。在RabbitMQ中,消息優先順序的實現方式是:在聲明queue時設置隊列的x-max-priority屬性,然後在publish消息時,設置消息的優先順序即可。 RabbitMQ優 ...


    一、引言

    在具體業務中可能會遇到一些要提前處理的消息,比如普通客戶的消息按先進先出的順序處理,Vip客戶的消息要提前處理。在RabbitMQ中,消息優先順序的實現方式是:在聲明queue時設置隊列的x-max-priority屬性,然後在publish消息時,設置消息的優先順序即可。

    RabbitMQ優先順序隊列註意事項:

    1)RabbitMQ3.5以後才支持優先順序隊列。

    2)只有當消費者不足,不能及時進行消費的情況下,優先順序隊列才會生效。

    3)優先順序取值範圍在0~9之間,數值越大則優先順序越高。

    二、示例

    2.1、發送端(生產端)

    新建一個控制台項目Send,並添加一個類RabbitMQConfig。

    class RabbitMQConfig
    {
        public static string Host { get; set; }

        public static string VirtualHost { get; set; }

        public static string UserName { get; set; }

        public static string Password { get; set; }

        public static int Port { get; set; }

        static RabbitMQConfig()
        {
            Host = "192.168.2.242";
            VirtualHost = "/";
            UserName = "hello";
            Password = "world";
            Port = 5672;
        }
    }
RabbitMQConfig.cs
    class Program
    {
        static void Main(string[] args)
        {
            Console.WriteLine("按任意鍵開始生產。");
            Console.ReadLine();
            PriorityMessagePublish();
            Console.ReadLine();
        }

        private static void PriorityMessagePublish()
        {
            const string MessagePrefix = "message_";
            const int PublishMessageCount = 6;
            byte messagePriority = 0;

            var factory = new ConnectionFactory()
            {
                HostName = RabbitMQConfig.Host,
                Port = RabbitMQConfig.Port,
                VirtualHost = RabbitMQConfig.VirtualHost,
                UserName = RabbitMQConfig.UserName,
                Password = RabbitMQConfig.Password,
                Protocol = Protocols.DefaultProtocol
            };

            using (var connection = factory.CreateConnection())
            {
                using (var channel = connection.CreateModel())
                {
                    //設置隊列優先順序,取值範圍在0~255之間。
                    Dictionary<string, object> dict = new Dictionary<string, object>
                    {
                        { "x-max-priority", 255 }
                    };

                    //聲明隊列
                    channel.QueueDeclare(queue: "priority", durable: true, exclusive: false, autoDelete: false, arguments: dict);


                    //向該消息隊列發送消息message
                    Random random = new Random();
                    for (int i = 0; i < PublishMessageCount; i++)
                    {
                        var properties = channel.CreateBasicProperties();
                        messagePriority = (byte)random.Next(0, 9);
                        properties.Priority = messagePriority;//設置消息優先順序,取值範圍在0~9之間。
                        var message = MessagePrefix + i.ToString();
                        var body = Encoding.UTF8.GetBytes(message);
                        channel.BasicPublish(exchange: "", routingKey: "priority", basicProperties: properties, body: body);
                        Console.WriteLine($"{DateTime.Now.ToString()} Send {message} , Priority {messagePriority}");
                    }
                }
            }
        }
    }
Program.cs

    2.2、接收端(消費端)

    新建一個控制台項目Receive,按住Alt鍵,將發送端RabbitMQConfig類拖一個快捷方式到Receive項目中。

    class Program
    {
        static void Main(string[] args)
        {
            Console.WriteLine("按任意鍵開始消費。");
            Console.ReadLine();
            PriorityMessageSubscribe();
        }

        public static void PriorityMessageSubscribe()
        {
            var factory = new ConnectionFactory()
            {
                HostName = RabbitMQConfig.Host,
                UserName = RabbitMQConfig.UserName,
                Password = RabbitMQConfig.Password
            };

            using (var connection = factory.CreateConnection())
            {
                using (var channel = connection.CreateModel())
                {
                    channel.BasicQos(prefetchSize: 0, prefetchCount: 1, global: false);
                    var consumer = new EventingBasicConsumer(channel);
                    consumer.Received += async (model, ea) =>
                    {
                        await Task.Run(() =>
                        {
                            var message = Encoding.UTF8.GetString(ea.Body);
                            Thread.Sleep(1000 * 2);
                            channel.BasicAck(deliveryTag: ea.DeliveryTag, multiple: false);//手動消息確認
                            Console.WriteLine($"{DateTime.Now.ToString()} Received {message}");
                        });
                    };
                    channel.BasicConsume(queue: "priority", noAck: false, consumer: consumer);//需要啟用消息響應,否則priority無效。
                    Console.ReadKey();
                }
            }
        }
    }
Program.cs

    2.3、運行結果

    從消費情況可以看出,message_2及message_3由於priority優先順序最高都是7,所以它們會被最早消費,而message_5的priority是0,所以最後才被消費。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、目標 搭建jenkins伺服器以及配置一臺.net編譯的slave客戶端 完整跑通一個asp.net項目的發佈 二、搭建jenkins環境 2.1 安裝jenkins 1. 根據官網提示安裝 進入jenkins官網下載==最新(如果不是最新的包,可能會存在推薦插件不能安裝的現象)==的穩定包,我 ...
  • 清華大學鏡像地址:https://mirrors.tuna.tsinghua.edu.cn/jenkins/updates/update-center.json操作步驟:進入jenkins系統管理進入插件管理點擊高級,修改升級站點的地址為清華大學鏡像地址二、更換源配置1. 關閉對 update-ce... ...
  • 本文接上篇文章 C#獲取設備(Audio和Video)名稱 簡單整理,對第四種方式使用整理. EnumDevice.dll是網上下載的,也下載了對應的源代碼, 對應dll:https://download.csdn.net/download/QQ81867376/12322158 該dll的源碼: ... ...
  • 直接上測試代碼和運行結果 static void Main(string[] args) { #region 測試代碼 List dataList; dataList = DirectXHelper.GetAudioDevicesList(); OutPutInfo("DirectX獲... ...
  • 一、引言 日常生活中,很多的APP都有延遲隊列的影子。比如在手機淘寶上,經常遇到APP派發的限時消費紅包,一般有幾個小時或24小時不等。假如在紅包倒計時的過程中,沒有消費掉紅包的話,紅包會自動失效。假如上述行為使用RabbitMQ延時隊列來理解的話,就是在你收到限時消費紅包的時候,手機淘寶會自動發一 ...
  • 最近在做微信公眾號項目,配置過程中發現,如果公眾號已設置了自定義菜單,如用戶點擊菜單後自動回覆文字、圖片、語音、視頻和圖文消息等。但如果該公眾號第三方開髮網站也需要用到自動回覆功能,則需要啟用:開發>>基本配置>>伺服器配置。啟用時會提示:開啟後,用戶發送的消息將自動轉發到該配置地址,並且在網站中設 ...
  • 知道為什麼要用內插字元串,只有踩過坑的人才能明白,如果你曾今使用string.format超5個以上占位符,那其中的痛苦我想你肯定是能夠共鳴的。 一:痛苦經歷 先上一段曾今寫過的一段代碼,大家來體會一下: 這裡擁有多達8個占位符,當年寫這個的時候,會有三個痛點。 1. 占位符不能寫錯了 比如這裡的 ...
  • .net core 集成 sentry 進行異常報警 Intro Sentry 是一個實時事件日誌記錄和彙集的平臺。其專註於錯誤監控以及提取一切事後處理所需信息而不依賴於麻煩的用戶反饋。它分為客戶端和服務端,客戶端(目前客戶端有 C , Python, PHP, JavaScript, Ruby等多 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...