MSMQ消息队列,包括远程访问
之前的项目用到了队列,现在总结一下,下面有非常详细的DEMO,希望能对有需要的人提供帮助。
使用场景:在项目中,将一些无需即时返回且耗时的操作提取出来,进行了异步处理,而这种异步处理的方式大大的节省了服务器的请求响应时间,从而提高了系统的吞吐量。
我的需求很简单,就是多个客户端连接到我的一个小型的数据转发服务器上,开始使用的是Socket通信实现这个功能,一旦数据服务器接收到来自不同客户端发来的消息,就对这些消息进行处理(我这里是将数据接收到后再转发到另一个服务器上),但考虑到客户端是每隔一个很短的时间周期向服务器发送信息,并且连接客服端数量比较多的时候,担心会产生并发访问的问题,也希望避免 数据转发服务器 频繁地从多个不同线程获取信息而出现其他未知问题,所以在处理客户端向数据转发服务器发送信息的时候采取队列的方式。
一般情况下,使用MSMQ,首先要安装消息服务,跟安装IIS一个套路,打开启用或关闭Windows功能窗口,找到并勾选MSMQ消息服务,然后点击确定进行安装,还不明白的百度一下;
在VS里添加 Messaging引用,就可以使用MessageQueue这个类了;接下来就要思考清楚你的数据(消息)的流向问题,之前因为自己对队列的错误认识,对到底在哪创建队列,队列的消息又由谁去发送和接收没有弄清除,还有参考的一些写得不是太清晰地博文,绕了好大一圈,所以今天在这里以我自己的项目需求为例子,说明 1、如何创建队列 2、如何向队列发送消息 3、 如何获取队列中的消息
首先、创建队列:根据我的需求,我要通过Socket通信将信息发送至数据转发服务器,因此为了避免并发访问问题的产生,消息队列应当建立在数据转发服务器上;
System.Messaging.MessageQueue myQuere = null; /// <summary> /// 这样就在数据转发服务器端创建了一个名为queuedemo的消息队列; /// 从客户端要发送的消息就保存在这个队列里, /// 你可以通过计算机管理->服务和应用下的消息队列中看到你创建的queuedemo队列, /// private$关键字是说明队列为专用队列, /// 如果没有这个关键字还要配置域服务器,还是挺麻烦,这个还是借助百度吧, /// 前面的“.”代表创建的队列目录是本机 /// </summary> /// <param name="sender"></param> /// <param name="e"></param> private void btnQueueInit_Click(object sender, EventArgs e) { // 注意这里是$符号 string queuePath = @".\private$\quereDemo"; // 判断消息队列示例是否存在 if (!System.Messaging.MessageQueue.Exists(queuePath)) { // 不存在则创建一个消息队列 myQuere = System.Messaging.MessageQueue.Create(queuePath); } myQuere = new System.Messaging.MessageQueue(queuePath); }
这样就在数据转发服务器端创建了一个名为queuedemo的消息队列;从客户端要发送的消息就保存在这个队列里,你可以通过计算机管理->服务和应用下的消息队列中看到你创建的queuedemo队列,private$关键字是说明队列为专用队列,如果没有这个关键字还要配置域服务器,还是挺麻烦,这个还是借助百度吧,前面的“.”代表创建的队列目录是本机,这个队列一旦创建成功,就是系统的事了,接下来要做的就是你怎么去把消息写进这个队列,或者读取队列的值
这里要特别注意,不要将queuepath路径字符串写成
string queuePath = @"FormatName:Direct=TCP:192.168.1.153\private$\quereDemo";
这样写的话是用于远程计算机对这个队列进行访问的,因为MessageQueue的Create()和Exisit()方法是没办法去识别上述FormatName格式的,还有要确保Create()函数要被执行了之后再用MessageQueue实例去引用;这样服务器端队列的创建就完成了;
在客户端中,向队列发送信息;
/// <summary> /// 写入数据到消息队列 /// </summary> /// <param name="sender"></param> /// <param name="e"></param> private void btnWriteMessage_Click(object sender, EventArgs e) { // 要往队列里写入的消息 // 要求发送的对象要以序列化的方式写进去,所以要设置formatter,这里用的是XmlMessageFormatter 还有BinaryMessageFormatter等等 string s = "客户端往队列里发送的消息"; // 实例化一个消息队列Object System.Messaging.Message message = new System.Messaging.Message(); message.Body = s; // body 为object类型 message.Formatter = new XmlMessageFormatter(new Type[] { typeof(string) }); // 选择xml的方式进行传送消息 // 创建输送消息的队列对象(在客户端和消息队列服务器分离的时候需要创建实例) // System.Messaging.MessageQueue myQuere = new System.Messaging.MessageQueue(@"FormatName:Direct=TCP:192.168.1.153\private$\queuedemo"); myQuere.Send(message); }
在客户端中,用一个MessageQueue实例指向服务器本机上创建的队列路径,这时,MessageQueue实例的构造函数里的路径就一定要用FormatName格式,指明是TCP通信还是HTTP还是Machine如我上面代码所示,然后调用Send()方法,将消息写进队列,这个要求发送的对象要以序列化的方式写进去,所以要设置formatter,这里用的是XmlMessageFormatter 还有BinaryMessageFormatter等等 注意保存你消息的 消息体Body是Object类型的 因此可以将你写的任何一个类的对象发送至消息队列
在服务器中接收消息队列
/// <summary> /// 开始读取消息 /// </summary> /// <param name="sender"></param> /// <param name="e"></param> private void btnReadMessage_Click(object sender, EventArgs e) { // 实例化消息队列 System.Messaging.MessageQueue msgQuere = new System.Messaging.MessageQueue(@".\private$\quereDemo"); // 指定写入客户端的序列化方式 msgQuere.Formatter = new XmlMessageFormatter(new Type[] { typeof(string) }); // 开启线程读取 // 此处也可以写成 Thread thread = new Thread(()=> { }); System.Threading.Thread thread = new System.Threading.Thread(new System.Threading.ThreadStart(() => { // 此处使用无限读取,读取完了马上while,也可以用一个时间来间隔 while (true) { //接收 System.Messaging.MessageQueue 引用的队列中可用的第一条消息。此调用是同步的,在有可用消息前,它将一直阻止当前线程的执行。 System.Messaging.Message msg = msgQuere.Receive(); if (null != msg) { MessageBox.Show(msg.Body.ToString()); } } })); // 启动线程 thread.IsBackground = true; // 设置为后台线程 thread.Start(); }
在本机上可以新创建一个队列实例指向本机的队列,然后按照之前约定的序列化格式反序列化消息体所以将新的队列实例的foarmatter属性赋值为发送时的formatter属性如代码所示,这个时候就直接用Receive()得到消息体,然后对消息体里的信息做处理,我这里是开启一个线程显示队列的消息,只要有新的消息写入,我就在消息框中输出
这个时候可能客户端无法向远程服务器成功发送消息,原因基本权限问题 服务器的消息队列的权限没有对未验证的客户端开放 你要在服务器队列里分配对应权限 如果你想读取队列的内容 还需要加系统变量
问题解决办法
1. 服务器端(dos:compmgmt.msc)
- 服务器上消息队列权限设置:给ANONYMOUS LOGON赋予所有权限;
- 修改服务器的注册表,允许非验证客户端访问
- 注册表新增HKLM\Software\Microsoft\MSMQ\Parameters\security\AllowNonauthenticatedRpc项,设置其DWORD值为1
- 注册表新增HKLM\Software\Microsoft\MSMQ\Parameters\security\NewRemoteReadServerDenyWorkgroupClient项,设置其DWORD值为1
MSMQ的安全访问控制说明参见:http://msdn.microsoft.com/en-us/library/4108f68e-80f5-40e1-b3df-b713cc4dff79(prot.20).aspx
这样客户端就可以读取服务器里的队列信息了 当然一般业务逻辑上不这么做 因为他只负责发送消息 ,综上,就是使用消息队列 跨服务器读写的 最基本的用法