目录

  1. 一:MSMQ的一些理论上的知识
  2. 二:队列类型(Queue Type)
  3. 三:安装消息队列
  4. 四:在C#中Messagequeue class
  5. 五:MSMQ-发送消息到远程专用队列
  6. 六:例子
 

一、在学习Messagequeue 类之前,首先介绍一下MSMQ的一些理论上的知识           MSMQ(MicroSoft Message Queue,微软消息队列)官方的解释是:在多个不同的应用之间实现相互通信的一种异步传输模式,相互通信的应用可以分布于同一台机器上,也可以分布于相连的网络空间中的任一位置。MSMQ通过发送和接受消息使得应用程序之间的通信变的更快和更可靠。      它的实现原理是:消息的发送者把自己想要发送的信息放入一个容器中(我们称之为Message),然后把它保存至一个系统公用空间的消息队列(Message Queue)中;本地或者是异地的消息接收程序再从该队列中取出发给它的消息进行处理。           在消息传递机制中,有两个比较重要的概念。一个是消息,一个是队列。消息是由通信的双方所需要传递的信息,说白了它可以是各式各样的媒体,如文本、声音、图象,在我们编程的时候,它一般是一个类的对象或字符串等,消息最终的理解方式,为消息传递的双方事先商定,这样做的好处是,一是相当于对数据进行了简单的加密,二则采用自己定义的格式可以节省通信的传递量。队列是发送和接收消息的公用存储空间,它可以存在于内存中或者是物理文件中。           采用MSMQ带来的好处是:由于是异步通信,无论是发送方还是接收方都不用等待对方返回成功消息,就可以执行余下的代码,因而大大地提高了事物处理的能力;当信息传送过程中,信息发送机制具有一定功能的故障恢复能力;MSMQ的消息传递机制使得消息通信的双方具有不同的物理平台成为可能。在微软的.net平台上利用其提供的MSMQ功能,可以轻松创建或者删除消息队列、发送或者接收消息、甚至于对消息队列进行管理。

二:队列类型(Queue Type)           由您或网络中的其他用户创建的队列和系统队列。用户创建的队列可能是以下任何一种队列:            “公共队列”在整个“消息队列”网络中复制,并且有可能由网络连接的所有站点访问。            “专用队列”不在整个网络中发布。相反,它们仅在所驻留的本地计算机上可用。专用队列只能由知道队列的完整路径名或标签的应用程序访问。            “管理队列”包含确认在给定“消息队列”网络中发送的消息回执的消息。指定希望 MessageQueue 组件使用的管理队列(如果有的话)。                  “响应队列”包含目标应用程序接收到消息时返回给发送应用程序的响应消息。指定希望 MessageQueue 组件使用的响应队列(如果有的话)。             系统生成的队列一般分为以下几类: “日记队列”可选地存储发送消息的副本和从队列中移除的消息副本。每个“消息队列”客户端上的单个日记队列存储从该计算机发送的消息副本。在服务器上为每个队列创建了一个单独的日记队列。此日记跟踪从该队列中移除的消息。 “死信队列”存储无法传递或已过期的消息的副本。如果过期或无法传递的消息是事务性消息,则被存储在一种特殊的死信队列中,称为“事务性死信队列”。死信存储在过期消息所在的计算机上。有关超时期限和过期消息的更多信息,请参见默认消息属性。 “报告队列”包含指示消息到达目标所经过的路由的消息,还可以包含测试消息。每台计算机上只能有一个报告队列。 “专用系统队列”是一系列存储系统执行消息处理操作所需的管理和通知消息的专用队列。 在应用程序中进行的大多数工作都涉及访问公共队列及其消息。但是,根据应用程序的日记记录、确认和其他特殊处理需要,在日常操作中很可能要使用几种不同的系统队列。

三:安装消息队列

如果你相使用消息队列进行通信的话,你就必须在你的电脑上安装消息队列,通过组件安装,具体安装的方法在这里就不详解了,类似于安装IIS一样

 

  对消息队列有了简单的了解后,使用MSMQ进行软件开发需要安装MSMQ,安装完后就该进入实际的开发阶段。具体的安装过程就是在控制面板里“添加/删除程序”下“添加/删除Windows组件”,完成添加就OK。安装完成后就可以通过交互界添加新的消息队列,详细如下图:

  

出了上面这种交互界面来创建MSMQ外,也可以通过编程来完成。

四:在C#中Messagequeue class

在使用.net开发msmq时,你必须引入命名空System.Messaging

MessageQueue 支持两种消息类型,同步和异步,同步方法使用的是peek();receive();异步使用的是:

Beginpeek() and Beginreceive();两者没有什么本质区别,都是封装好的方法,你只要直接创建Messagequeue的对象来调用这个方法就OK了, 我想至于什么同步和异步是什么意思,就没有必有解释了吧,

五:MSMQ-发送消息到远程专用队列

在工作组模式下,远程访问专用队列。在网上找到一篇文章,翻译了一下。
最后结论,直接使用多元素格式名方式,利用IP地址直接对单个或多个目标发送消息
MessageQueue rmQ = new MessageQueue("FormatName:Direct=TCP:121.0.0.1//private$//queue,Direct=TCP:192.168.1.2//private$//queue");
rmQ.Send("sent to regular queue - Atul");
原文地址
http://www.infosysblogs.com/microsoft/2007/02/msmq_sending_message_to_remote.html
 
1.当需要引用远程队列时,使用“machinename/private$/queuename”的格式无法工作。会返回“invalid queue path”错误。
 
 
2.队列名称需要使用“"FormatName:Direct=OS:machinename//private$//queuename”的格式。其他友好形式的表达式都是被转换为FormatName格式之后进行调用的。而且其他这些友好表达式的转换是基于Active Directory(域)来进行解析的,如果没有域的支持,这些表达式将无法工作。
 
例:
    MessageQueue rmQ = new MessageQueue 
                                    ("FormatName:Direct=OS:machinename//private$//queue");
    rmQ.Send("sent to regular queue - Atul");
 
 
3.FontName是区分大小写的。如果表达式为“FORMATNAME:Direct=OS:machinename//private$//queuename”,是无法工作的。但这种表达式不会返回任何错误。FontName好像是表达式里唯一区分大小写的部分。其他部分可以随意使用大小写,例如可以使用“DIRECT”。
 
 
4.如果想要使用机器IP地址,表达式的语法为“FormatName:Direct=TCP:ipaddress//private$//queuename”
 
  例:
     MessageQueue rmQ = new MessageQueue
                                     ("FormatName:Direct=TCP:121.0.0.1//private$//queue");
     rmQ.Send("sent to regular queue - Atul");
 
 
5.在代码中创建的队列实例对象的事务性属性,必须与要发送的目标队列的属性相匹配。前面的例子中发送的消息为非事务型消息,如果要发送消息到事务型的队列,代码为:
    MessageQueue rmTxnQ = new MessageQueue
                                            ("FormatName:Direct=OS:machinename//private$//queue");
    rmTxnQ.Send("sent to Txn queue - Atul", MessageQueueTransactionType.Single);
 
如果事务型属性不匹配,消息将无法传递。系统不会返回任何错误,但该条消息却会丢掉。
 
 
6.最后,当你发送消息到远程队列,系统会在本机创建一个临时的传出队列。这样做的目的是防止远程队列无法访问。在计算机管理器中查看消息队列/传出队列,可以看到这些临时队列。在管理器的右侧可以显示状态(联通、未联通)以及IP地址。

六:例子

------------------------------------------------------------------------------

using System;

using System.Collections.Generic;

using System.Text;

using System.Messaging;

namespace MyQueue

{

public class MyMessageQueue

{

private string Path;

/// <summary>

/// 1.通过Create方法创建使用指定路径的新消息队列

/// </summary>

/// <param name="queuePath"></param>

public void Createqueue(string queuePath)

{

try

{

if (!MessageQueue.Exists(queuePath))

{

MessageQueue.Create(queuePath);

}

else

{

Console.WriteLine(queuePath + "已经存在!");

//MessageQueue.Delete(queuePath);

//MessageQueue.Create(queuePath);

//Console.WriteLine(queuePath + "删除重建");

}

Path = queuePath;

}

catch (MessageQueueException e)

{

Console.WriteLine(e.Message);

}

}

/// <summary>

///  2.连接消息队列并发送消息到队列

/// 远程模式:MessageQueue rmQ = new MessageQueue("FormatName:Direct=OS:machinename//private$//queue");

///     rmQ.Send("sent to regular queue - Atul");对于外网的MSMQ只能发不能收

/// </summary>

public void SendMessage()

{

try

{

//连接到本地队列

MessageQueue myQueue = new MessageQueue(Path);

//MessageQueue myQueue = new MessageQueue("FormatName:Direct=TCP:192.168.12.79//Private$//myQueue1");

//MessageQueue rmQ = new MessageQueue("FormatName:Direct=TCP:121.0.0.1//private$//queue");--远程格式

Message myMessage = new Message();

myMessage.Body = "消息内容34kuangbo去死";

myMessage.Formatter = new XmlMessageFormatter(new Type[] { typeof(string) });

//发生消息到队列中

myQueue.Send(myMessage);

Console.WriteLine("消息发送成功!");

Console.ReadLine();

}

catch (ArgumentException e)

{

Console.WriteLine(e.Message);

}

}

/// <summary>

/// 3.连接消息队列并从队列中接收消息

/// </summary>

public void ReceiveMessage()

{

MessageQueue myQueue = new MessageQueue(Path);

myQueue.Formatter = new XmlMessageFormatter(new Type[] { typeof(string) });

try

{

//从队列中接收消息

Message myMessage = myQueue.Receive();// myQueue.Peek();--接收后不消息从队列中移除

string context = myMessage.Body.ToString();

Console.WriteLine("消息内容:" + context);

Console.ReadLine();

}

catch (MessageQueueException e)

{

Console.WriteLine(e.Message);

}

catch (InvalidCastException e)

{

Console.WriteLine(e.Message);

}

}

/// <summary>

/// 4.清空指定队列的消息

/// </summary>

public void ClealMessage()

{

MessageQueue myQueue = new MessageQueue(Path);

myQueue.Purge();

Console.WriteLine("已清空对了{0}上的所有消息",Path);

}

/// <summary>

/// 5.连接队列并获取队列的全部消息

/// </summary>

public void GetAllMessage()

{

MessageQueue myQueue = new MessageQueue(Path);

Message[] allMessage = myQueue.GetAllMessages();

XmlMessageFormatter formatter = new XmlMessageFormatter(new Type[] { typeof(string) });

for (int i = 0; i < allMessage.Length; i++)

{

allMessage[i].Formatter = formatter;

Console.WriteLine("第{0}机密消息为:{1}", i+1, allMessage[i].Body.ToString());

}

Console.ReadLine();

}

}

}

------------------------
using System;
using System.Collections.Generic;
using System.Text;
namespace MyQueue
{
    class Program
    {
        static void Main(string[] args)
        {
            MyMessageQueue queue = new MyMessageQueue();
            queue.Createqueue(".//Private$//myQueue2");
            queue.SendMessage();
            queue.GetAllMessage();
            //queue.ReceiveMessage();
            //queue.ClealMessage();
        }
    }
}

注意:传输的 路径格式 要正确。

出处:http://blog.csdn.net/sage425/article/details/6298461

MSMQ-发送消息到远程专用队列 实例的更多相关文章

  1. MSMQ-发送消息到远程专用队列path格式

    在工作组模式下,远程访问专用队列.在网上找到一篇文章,翻译了一下. 最后结论,直接使用多元素格式名方式,利用IP地址直接对单个或多个目标发送消息      MessageQueue rmQ = new ...

  2. RocketMQ3.2.2生产者发送消息自动创建Topic队列数无法超过4个

    问题现象 RocketMQ3.2.2版本,测试时尝试发送消息时自动创建Topic,设置了队列数量为8: producer.setDefaultTopicQueueNums(8); 同时设置broker ...

  3. 深入delphi编程理解之消息(三)发送消息函数的一般应用实例

    通过对消息函数(SendMessage.PostMessage.Perform)的一般应用,来说明sendmessage.postmessage函数和perform 方法调用方式和结果的区别. 一.程 ...

  4. Delphi实现获取句柄并发送消息的方法(FindWindow、FindWindowEx、EnumChildWindows、SendMessage)

    Delphi实现获取句柄并发送消息的方法 本文以实例形式详细说明了Delphi获取句柄并发送消息的方法,具体用法说明如下: 查找另外一个窗口的句柄: handle := FindWindow(nil, ...

  5. MSMQ向远程服务器发送消息----错误总结

    一:路径错误(Path)错误 如果向远程服务器发送消息,请使用格式名的形式,如: FormatName:Direct=TCP:121.0.0.1\\private$\\queueFormatName: ...

  6. 【6】.net msmq消息队列实例

    1.msmq消息队列windows环境安装 控制面板---->程序和功能---->启用或关闭Windows程序---->Microsoft Message Queue(MSMQ)服务 ...

  7. 【c#】队列(Queue)和MSMQ(消息队列)的基础使用

    首先我们知道队列是先进先出的机制,所以在处理并发是个不错的选择.然后就写两个队列的简单应用. Queue 命名空间 命名空间:System.Collections,不在这里做过多的理论解释,这个东西非 ...

  8. 【转】MSMQ 微软消息队列 简单 示例

    MSMQ它的实现原理是:消息的发送者把自己想要发送的信息放入一个容器中(我们称之为Message),然后把它保存至一个系统公用空间的消息队列(Message Queue)中:本地或者是异地的消息接收程 ...

  9. System V 消息队列 实例

    前言: 消息队列是消息的链接表,存放在内核中,并由消息队列标识符标识.我们将称消息队列为 “队列”,其标识符为“队列I D”.msgget创建一个新队列或打开一个存在的队列; msgsnd向队列末端添 ...

随机推荐

  1. web报告工具FineReport在使用方法和解决方案常见错误遇到(一)

    FineReport在使用方法和解决方案常见错误遇到(一) 这里写的开胃菜.我希望我们能理清自己的问题和解决办法干出来的,Mark一点点.有利于所有. 失败搜索出,如果有一个文件,看看你的度娘那里.看 ...

  2. SRM 590 DIV1

    转载请注明出处,谢谢viewmode=contents">http://blog.csdn.net/ACM_cxlove?viewmode=contents    by---cxlov ...

  3. Eclipse建筑物SSH(struts-2.2.3 + spring-2.5.6 + hibernate-3.6.8)相框-随着源代码

    一直想自己搭建一个ssh框架,这次因为编写demo的须要,就亲手搭建了一下,并逐步測试!以下进入正题: 创建Struts项目 整合步骤: 1,在Eclipse中创建一个DynamicWeb Proje ...

  4. Windows 8 键盘上推自定义处理

    原文:Windows 8 键盘上推自定义处理 在Windows 8 应用程序中,当TextBox控件获得焦点时,输入面板会弹出,如果TextBox控件处于页面下半部分,则系统会将页面上推是的TextB ...

  5. 【Java基础】System.arraycopy()的使用详解

    由于在Java中System.arraycopy()方法在一维数组和二维数组中的表现不同,所以做了一个测试 public static void main(String[] args) { int[] ...

  6. 【SSH进阶之路】一步步重构容器实现Spring框架——彻底封装,实现简单灵活的Spring框架(十一)

    文件夹      [SSH进阶之路]一步步重构容器实现Spring框架--从一个简单的容器開始(八)      [SSH进阶之路]一步步重构容器实现Spring框架--解决容器对组件的"侵入 ...

  7. 在高德地图应用api,和api展出的标记小的应用程序

    <!DOCTYPE HTML> <html> <head> <meta http-equiv="Content-Type" content ...

  8. 小牟Andorid下面MD5具体实现的思路总结

    Android的开发往往需要一定数目demo 从今起MD5一些加密算法提取物 看看是如何实现的 首先,我们必须明确为什么加密? 1 数据安全处理 2 防止数据窃取 3 有效的避免恶意攻击 4 保证文件 ...

  9. MEF初体验之八:过滤目录

    当在使用子容器的时候,基于某些具体标准来过滤目录可能是重要的.例如,基于部件的创建策略来过滤是很常见的.下面的代码片段演示了如何构建这种特别方法: var catalog = new Assembly ...

  10. 实现android里面WebView显示内容

    在日常学习.我们会看到,当手机应该检查注册协议.单击协议时,以查看内部协议的全部内容! 以下给大家看看实现的过程: 首先贴show_xy.XML代码: <WebView android:layo ...