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
这样客户端就可以读取服务器里的队列信息了 当然一般业务逻辑上不这么做 因为他只负责发送消息 ,综上,就是使用消息队列 跨服务器读写的 最基本的用法
MSMQ消息队列,包括远程访问的更多相关文章
- 微软MSMQ消息队列的使用
首先在windows系统中安装MSMQ 一.MSMQ交互 开发基于消息的应用程序从队列开始.MSMQ包含四种队列类型: 外发队列:消息发送到目的地之前,用它来临时存储消息. 公共队列:在主动目录中公布 ...
- 【转】MSMQ消息队列安装
一.Windows 7安装.管理消息队列1.安装消息队列 执行用户必须要有本地 Administrators 组中的成员身份,或等效身份. 具体步骤: 开始—>控制面板—>程 ...
- MSMQ消息队列安装
一.Windows 7安装.管理消息队列1.安装消息队列 执行用户必须要有本地 Administrators 组中的成员身份,或等效身份. 具体步骤: 开始—>控制面板—>程 ...
- 【6】.net msmq消息队列实例
1.msmq消息队列windows环境安装 控制面板---->程序和功能---->启用或关闭Windows程序---->Microsoft Message Queue(MSMQ)服务 ...
- MSMQ消息队列 用法
引言 接下来的三篇文章是讨论有关企业分布式开发的文章,这三篇文章筹划了很长时间,文章的技术并不算新,但是文章中使用到的技术都是经过笔者研究实践后总结的,正所谓站在巨人的肩膀上,笔者并不是巨人,但也希望 ...
- 跟我一起学WCF(1)——MSMQ消息队列
一.引言 Windows Communication Foundation(WCF)是Microsoft为构建面向服务的应用程序而提供的统一编程模型,该服务模型提供了支持松散耦合和版本管理的序列化功能 ...
- C#实战Microsoft Messaging Queue(MSMQ)消息队列(干货)
前言 在使用MSMQ之前,我们需要自行安装消息队列组件!(具体安装方法大家自己搜一下吧) 采用MSMQ带来的好处是:由于是异步通信,无论是发送方还是接收方都不用等待对方返回成功消息,就可以执行余下的代 ...
- C#实战Microsoft Messaging Queue(MSMQ)消息队列
前言 在使用MSMQ之前,我们需要自行安装消息队列组件!(具体安装方法大家自己搜一下吧) 采用MSMQ带来的好处是:由于是异步通信,无论是发送方还是接收方都不用等待对方返回成功消息,就可以执行余下的代 ...
- C#实战Microsoft Messaging Queue(MSMQ)消息队列(干货)<转>
前言 在使用MSMQ之前,我们需要自行安装消息队列组件!(具体安装方法大家自己搜一下吧) 采用MSMQ带来的好处是:由于是异步通信,无论是发送方还是接收方都不用等待对方返回成功消息,就可以执行余下的代 ...
随机推荐
- Visual Studio IronPython CPython
安装 IronPython - 张善友 - 博客园https://www.cnblogs.com/shanyou/archive/2006/09/14/504580.html VS2017作为pyth ...
- 为什么vue组件的属性,有的需要加冒号“:”,有的不用?
https://segmentfault.com/q/1010000010929963/a-1020000010930077 <tab :line-width="2" act ...
- Python之Django之views中视图代码重复查询的优化
Django框架中views视图中如果多个函数都有同样的查询语句,例如: allcategory = Category.objects.all() remen = Article.objects.fi ...
- zabbix使用fping监控任意两个节点之间的网络质量、丢包率和响应时间
zabbix使用fping监控任意两个节点之间的网络质量.丢包率和响应时间 之前的博文 使用zabbix3..4的ICMP Ping模版实现对客户端网络状态的监控 https://www.cnblog ...
- 零基础学Python-第一章 :Python介绍和安装-01.Python语言的特点
结束
- Source Insight解决回车缩进过多问题
摘自:https://jingyan.baidu.com/article/02027811339e651bcc9ce53c.html 新安装的Source Insight软件在编写代码时,回车换行 ...
- ORA-02287: sequence number not allowed here问题的解决
当插入值需要从另外一张表中检索得到的时候,如下语法的sql语句已经不能完成该功能:insert into my_table(id, name) values ((select seq_my_table ...
- CentOS7下yum安装Redis
(1).Redis概述 Redis是一个开源的使用ANSI C语言编写.支持网络.可基于内存亦可持久化的日志型.Key-Value(键值型)数据库(非关系型数据库),并提供多种语言的API. Redi ...
- 今天被这个BDE错误搞了半天,不过终于好了,分享一下
今天正编译程序时,突然就报了这个错误出来,重启电脑都没用,多亏网上高手指教,先把解决方案列于下,供受此累得朋友查阅,自己也留底供查找:"Shared memory conflict ($21 ...
- 机器学习之径向基神经网络(RBF NN)
本文基于台大机器学习技法系列课程进行的笔记总结. 主要内容如下图所示: 首先介绍一下径向基函数网络的Hypothesis和网络的结构,然后介绍径向基神经网络学习算法,以及利用K-means进行的学习, ...