C# 消息隊(duì)列(消息隊(duì)列的編程實(shí)現(xiàn))
當(dāng)前位置:點(diǎn)晴教程→知識(shí)管理交流
→『 技術(shù)文檔交流 』
? 消息隊(duì)列的編程實(shí)現(xiàn)既然理解了消息隊(duì)列的體系結(jié)構(gòu)之后,就可以探討其編程了。下面幾節(jié)將學(xué)習(xí)如何創(chuàng)建和控制 隊(duì)列,如何發(fā)送和接收消息。 還要構(gòu)建一個(gè)小型課程訂單應(yīng)用程序,它由發(fā)送部分和接收部分組成。 創(chuàng)建消息隊(duì)列創(chuàng)建消息隊(duì)列 前面了解了如何使用Computer Management 實(shí)用程序創(chuàng)建消息隊(duì)列。消息隊(duì)列還可以用 MessageQueue 類的Create()方法以編程方式創(chuàng)建。 調(diào)用 Create()方法之后,就可以修改隊(duì)列的屬性。例如,使用Label 屬性,把隊(duì)列的標(biāo)簽設(shè)置為 Demo Queue。示例程序把隊(duì)列的路徑和格式名寫到控制臺(tái)上。格式名用UUID 自動(dòng)創(chuàng)建,UUID 可 用于訪問(wèn)隊(duì)列,且無(wú)須服務(wù)器名:
創(chuàng)建隊(duì)列時(shí)需要管理權(quán)限。通常不希望應(yīng)用程序的用戶擁有管理權(quán)限。這就是隊(duì) 列通常用安裝程序創(chuàng)建原因. 查找隊(duì)列路徑名和格式名可以用于標(biāo)識(shí)隊(duì)列。要查找隊(duì)列,必須區(qū)分公共隊(duì)列和私有隊(duì)列。公共隊(duì)列在 Active Directory 中發(fā)布。對(duì)于這些隊(duì)列,無(wú)須知道它們所在的系統(tǒng)。只有在己知隊(duì)列所在系統(tǒng)私有 隊(duì)列名稱時(shí)才能找到私有隊(duì)列。
GetPublicQueues()方法是重載的。它的一個(gè)重載版本允許傳遞MessageQueueCriteria 類的一個(gè)實(shí) 例。利用這個(gè)類可以搜索在某個(gè)時(shí)刻之前或之后創(chuàng)建或修改的隊(duì)列,還可以查找隊(duì)列的類別、標(biāo)簽 或計(jì)算機(jī)名。 可以使用靜態(tài)方法GetPrivateQueuesByMachine()搜索私有隊(duì)列。這個(gè)方法返回指定系統(tǒng)中的所 有私有隊(duì)列。 打開(kāi)已知隊(duì)列如果隊(duì)列名已知,就不需要搜索它。使用路徑或格式名就可以打開(kāi)隊(duì)列。路徑或格式名都在 MessageQueue 類的構(gòu)造函數(shù)中設(shè)置。 路徑名路徑指定了打開(kāi)隊(duì)列需要的計(jì)算機(jī)名和隊(duì)列名。下面的代碼示例打開(kāi)本地主機(jī)上的 MyPublicQueue 隊(duì)列。為了確定隊(duì)列是否存在,可以使用靜態(tài)方法MessageQueue.Exists():
根據(jù)隊(duì)列的類型,在打開(kāi)隊(duì)列時(shí)需要不同的標(biāo)識(shí)符。下表 列出了指定類型的隊(duì)列名的語(yǔ)法。 在使用路徑名打開(kāi)公共隊(duì)列時(shí),需要傳遞計(jì)算機(jī)名。如果計(jì)算機(jī)名未知,則可以使用格式名代 替。私有隊(duì)列的路徑名只能在本地系統(tǒng)上使用,必須使用格式名遠(yuǎn)程訪問(wèn)私有隊(duì)列。 格式名除了路徑名之外,還可以使用格式名打開(kāi)隊(duì)列。格式名用于在Active Directory 中搜索隊(duì)列,獲 得隊(duì)列所在的主機(jī)。在斷開(kāi)連接的環(huán)境下,在發(fā)送消息時(shí)隊(duì)列不能到達(dá),此時(shí)就需要使用格式名:
格式名還有一些其他用途。它可以用于打開(kāi)私有隊(duì)列,并指定要使用的協(xié)議: FormatName:DIRECT=OS:MachineName\QueueName 是使用格式名指定隊(duì)列的另一種方式。 此時(shí)不需要指定協(xié)議,但仍可以使用計(jì)算機(jī)名和格式名。 發(fā)送消息可以使用MessageQueue 類的Send()方法給隊(duì)列發(fā)送消息。作為參數(shù)傳遞給Send()方法的對(duì)象序 列化到相關(guān)聯(lián)的隊(duì)列上。Send()方法是重載的,這樣才能傳遞標(biāo)簽和MessageQueueTransaction 對(duì)象。 Message Queuing 的事務(wù)行為在后面論述。 下面的代碼示例先檢查隊(duì)列是否存在,如果不存在,就創(chuàng)建一個(gè)隊(duì)列。接著打開(kāi)隊(duì)列,使用Send() 方法給隊(duì)列發(fā)送Sample Message 消息。 路徑名給服務(wù)器名指定“.”,表示它是本地系統(tǒng)。私有隊(duì)列的路徑名只能在本地使用。
消息格式化程序消息傳輸給隊(duì)列的格式取決于格式化程序。MessageQueue 類有一個(gè)Formatter 屬性,通過(guò)它可 以指定格式化程序。默認(rèn)的格式化程序XmlMessageFormatter 會(huì)用XML 語(yǔ)法格式化消息,如前面的 例子所示。 消息格式化程序?qū)崿F(xiàn)IMessageFormatter 接口。System.Messaging 名稱空間中有3 個(gè)消息格式化 程序: XmlMessageFormatter 是默認(rèn)的格式化程序,它使用XML 序列化對(duì)象,XML 格式的內(nèi)容詳。 使用 BinaryMessageFormatter,可以用二進(jìn)制格式對(duì)消息進(jìn)行序列化。這些消息比使用XML 格式化的消息短。 ActiveXMessageFormatter 是一個(gè)二進(jìn)制格式化程序,這樣可以用COM 對(duì)象讀寫消息。使 用這個(gè)格式化程序,可以用.NET 類把消息寫入隊(duì)列中,使用COM 對(duì)象從隊(duì)列中讀取消息, 反之亦然。 發(fā)送復(fù)雜的消息除了傳遞字符串之外,還可以給MessageQueue 類的Send()方法傳遞對(duì)象。雖然該類的類型必 須滿足一些特定的要求,但它們?nèi)Q于格式化程序。 對(duì)于二進(jìn)制格式化程序,該類必須用[Serializable]屬性序列化。使用.NET 運(yùn)行庫(kù)的序列化功能, 序列化所有字段(包括私有字段)。實(shí)現(xiàn)ISerializable 接口就可以定義自定義序列化。 XML 序列化在使用XML 格式化程序時(shí)進(jìn)行。在XML 序列化過(guò)程中,會(huì)序列化所有公共字段 和屬性。使用System.Xml.Serialization 名稱空間中的屬性可以影響XML 序列化。 接收消息要讀取消息,也可以使用MessageQueue 類。通過(guò)Receive()方法可以讀取一條消息,再將該消 息從隊(duì)列中刪除。如果使用不同的優(yōu)先級(jí)發(fā)送消息,就讀取優(yōu)先級(jí)最高的消息。讀取優(yōu)先級(jí)相同的 消息時(shí),第一條發(fā)送的消息不一定是第一條讀取的消息,因?yàn)橄⒃诰W(wǎng)絡(luò)中的傳遞順序無(wú)法保證。 要保證發(fā)送順序和讀取順序相同,可以使用事務(wù)消息隊(duì)列。 在下面的例子中,要從私有隊(duì)列MyPrivateQueue 中讀取一條消息。之前把一個(gè)簡(jiǎn)單的字符串傳 遞給該消息。在使用XmlMessageFormatter 格式化程序讀取消息時(shí),必須把要讀取的對(duì)象的類型傳 遞給該格式化程序的構(gòu)造函數(shù)。在本例中,將System.String 類型傳遞給XmlMessageFormatter 的構(gòu) 造函數(shù)的參數(shù)數(shù)組。這個(gè)構(gòu)造函數(shù)可以接收一個(gè)String 數(shù)組,該數(shù)組包含要作為字符串傳遞的類型; 也可以接收一個(gè)Type 數(shù)組。 用 Receive()方法讀取消息,再把消息正文寫入控制臺(tái)中:
Receive()方法將同步執(zhí)行,如果隊(duì)列中沒(méi)有消息,它就會(huì)等待隊(duì)列中有消息時(shí)再執(zhí)行。 枚舉消息除了使用 Receive()方法逐條消息地讀取之外,還可以使用枚舉器遍歷所有消息。因?yàn)?MessageQueue 類實(shí)現(xiàn)IEnumerable 接口,所以可以在foreach 語(yǔ)句中使用。使用迭代器時(shí),雖然消息 不會(huì)從隊(duì)列中刪除,但可以查看消息從而獲得它們的內(nèi)容:
除了使用IEnumerable 接口外,還可以使用MessageEnumerator 類。雖然MessageEnumerator 類實(shí)現(xiàn) IEnumerator 接口,但它有更多功能。實(shí)現(xiàn)IEnumerable 接口,就表示不從隊(duì)列中刪除消息。 MessageEnumerator 類的RemoveCurrent()方法可以從枚舉器的當(dāng)前光標(biāo)位置刪除消息。 在下面的例子中,使用MessageQueue 類的GetMessageEnumerator()方法訪問(wèn)MessageEnumerator 類。通過(guò)MessageEnumerator 類的MoveNext()方法,可以逐條查看消息。MoveNext()方法重載為允 許把一個(gè)時(shí)間段作為參數(shù)。這是使用這個(gè)枚舉器的一個(gè)主要優(yōu)點(diǎn)?,F(xiàn)在,線程可以在指定的時(shí)間段 內(nèi)等待消息到達(dá)隊(duì)列,之后就不等待了。IEnumerator 接口定義的Current 屬性返回消息的一個(gè)引用:
異步讀取MessageQueue 類的Receive()方法會(huì)等到隊(duì)列中的消息可以讀取為止。為了避免阻礙線程的執(zhí) 行,可以在Receive()方法的一個(gè)重載版本中指定一個(gè)超時(shí)期限。要在超時(shí)后讀取隊(duì)列中的消息,必 須再次調(diào)用Receive()方法。除了輪詢消息外,還可以調(diào)用BeginReceive()異步方法。在使用 BeginReceive()開(kāi)始異步讀取消息之前,應(yīng)設(shè)置ReceiveCompleted 事件。ReceiveCompleted 事件需要 ReceiveCompletedEventHandler 委托,在消息到達(dá)隊(duì)列并可以讀取時(shí)該委托引用要調(diào)用的方法。在下 面的例子中,把MessageArrived()方法傳遞給ReceivedCompletedEventHandler 委托:
MessageArrived()處理程序方法需要兩個(gè)參數(shù)。第一個(gè)參數(shù)是MessageQueue 事件源。第二個(gè)參 數(shù)是ReceiveCompletedEventArgs 類型,它包含消息和異步結(jié)果。在下面的例子中,調(diào)用隊(duì)列中的 EndReceive()方法,以獲得異步方法的結(jié)果,即消息:
閱讀原文:原文鏈接 該文章在 2025/4/24 10:02:21 編輯過(guò) |
關(guān)鍵字查詢
相關(guān)文章
正在查詢... |