NET中RabbitMQ的使用

https://www.cnblogs.com/xibei666/p/5931267.html

概述

  MQ全称为Message Queue, 消息队列(MQ)是一种应用程序对应用程序的通信方法。RabbitMQ是一个在AMQP基础上完整的,可复用的企业消息系统。他遵循Mozilla Public License开源协议。AMQP(高级消息队列协议) 是一个异步消息传递所使用的应用层协议规范,作为线路层协议,而不是API(例如JMS),AMQP 客户端能够无视消息的来源任意发送和接受信息。AMQP的原始用途只是为金融界提供一个可以彼此协作的消息协议,而现在的目标则是为通用消息队列架构提供通用构建工具。因此,面向消息的中间件 (MOM)系统,例如发布/订阅队列,没有作为基本元素实现。AMQP当中有四个概念非常重要(一个虚拟主机持有一组交换机、队列和绑定):

virtual host,虚拟主机

exchange,交换机

queue,队列

binding,绑定

  更多理论性东西可以参考(在Windows上安装Rabbit MQ 指南),针对队列的讲解相当详细

Window下安装RabbbitMQ

文件下载安装

Rabbit MQ 是建立在强大的Erlang OTP平台上,因此安装Rabbit MQ的前提是安装Erlang。通过下面两个连接下载安装3.2.3 版本:

下载并安装 Erlang OTP For Windows (vR16B03)

运行安装 Rabbit MQ Server Windows Installer (v3.2.3)

默认安装的Rabbit MQ 监听端口是5672。先安装Erlang OTP后安装RabbitMQ,安装方式默认即可,RabbitMQ可以勾选安装后台服务、服务启动和停止等操作。

激活Rabbit MQ's Management Plugin

使用Rabbit MQ 管理插件,可以更好的可视化方式查看Rabbit MQ 服务器实例的状态,打开CMD命令,cd到安装目录(..\rabbitmq_server-3.2.3\sbin)下,输入下面的命令激活:

rabbitmq-plugins enable rabbitmq_management

要重启服务才能生效,可以执行

net stop RabbitMQ && net start RabbitMQ

输入网址,打开监控页面: http://localhost:15672 (默认账号和密码:guest 和guest)

配置RabbitMQ用户权限

RabbitMQ是存在用户权限的,默认是guest 密码也是guest,隶属于Administrator管理员下。现需要配置新用户和权限,继续打开CMD命令,cd到安装目录sbin下:

用户操作指令:

复制代码

::查询服务状态

rabbitmqctl status

::列举虚拟主机列表

rabbitmqctl list_vhosts

::列举用户列表

rabbitmqctl list_users

:: 添加用户和密码

rabbitmqctl add_user hao abc123

:: 设置权限

rabbitmqctl set_permissions yy "." "." ".*"

:: 分配用户组

rabbitmqctl set_user_tags yy administrator

:: 删除guest用户

rabbitmqctl delete_user guest

::修改用户密码

rabbitmqctl change_password {username} {newpassowrd}

复制代码

.NET中RabbitMQ使用

1、Nuget下载RabbitMQ.Client第三方类库,版本V3.6.5,高版本与.NET Framework 4.5有冲突,RabbitMQ Client地址

2、利用RabbitMQ Clinet类库编码(代码内容有注释,此处不做详细解释,文章后有完整代码的下载地址)

  <1>RabbitMQ的direct类型Exchange

   Producter发送消息代码:

复制代码

///



/// 连接配置

///

private static readonly ConnectionFactory rabbitMqFactory = new ConnectionFactory(){

HostName ="192.168.1.8",UserName="hao",Password="abc123",Port= 5672

};

///



/// 路由名称

///

const string ExchangeName = "justin.exchange";

  1. //队列名称
  2. const string QueueName = "justin.queue";
  3. public static void DirectExchangeSendMsg()
  4. {
  5. using (IConnection conn = rabbitMqFactory.CreateConnection())
  6. {
  7. using (IModel channel = conn.CreateModel())
  8. {
  9. channel.ExchangeDeclare(ExchangeName, "direct", durable: true, autoDelete: false, arguments: null);
  10. channel.QueueDeclare(QueueName, durable: true, autoDelete: false, exclusive: false, arguments: null);
  11. channel.QueueBind(QueueName, ExchangeName, routingKey: QueueName);
  12. var props = channel.CreateBasicProperties();
  13. props.Persistent = true;
  14. string vadata = Console.ReadLine();
  15. while (vadata != "exit")
  16. {
  17. var msgBody = Encoding.UTF8.GetBytes(vadata);
  18. channel.BasicPublish(exchange: ExchangeName, routingKey: QueueName, basicProperties: props, body: msgBody);
  19. Console.WriteLine(string.Format("***发送时间:{0},发送完成,输入exit退出消息发送",DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss")));
  20. vadata = Console.ReadLine();
  21. }
  22. }
  23. }
  24. }

复制代码

   

  Customer接收消息代码:

复制代码

///



/// 连接配置

///

private static readonly ConnectionFactory rabbitMqFactory = new ConnectionFactory() {

HostName = "192.168.1.8", UserName = "hao", Password = "abc123", Port = 5672

};

///



/// 路由名称

///

const string ExchangeName = "justin.exchange";

  1. //队列名称
  2. const string QueueName = "justin.queue";
  3. public static void DirectAcceptExchange()
  4. {
  5. using (IConnection conn = rabbitMqFactory.CreateConnection())
  6. {
  7. using (IModel channel = conn.CreateModel())
  8. {
  9. channel.ExchangeDeclare(ExchangeName, "direct", durable: true, autoDelete: false, arguments: null);
  10. channel.QueueDeclare(QueueName, durable: true, autoDelete: false, exclusive: false, arguments: null);
  11. channel.QueueBind(QueueName, ExchangeName, routingKey: QueueName);
  12. while (true)
  13. {
  14. BasicGetResult msgResponse = channel.BasicGet(QueueName, noAck: true);
  15. if (msgResponse != null)
  16. {
  17. var msgBody = Encoding.UTF8.GetString(msgResponse.Body);
  18. Console.WriteLine(string.Format("***接收时间:{0},消息内容:{1}", DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"),msgBody));
  19. }
  20. //BasicGetResult msgResponse2 = channel.BasicGet(QueueName, noAck: false);
  21. ////process message ...
  22. //channel.BasicAck(msgResponse2.DeliveryTag, multiple: false);
  23. System.Threading.Thread.Sleep(TimeSpan.FromSeconds(1));
  24. }
  25. }
  26. }
  27. }

复制代码

  

  但是这种处理速度较慢,因为循环线程等待。高效的接收消息的方式可以使用EventingBasicConsumer进行消息接收处理,修改代码内容如下:

复制代码

public static void DirectAcceptExchangeEvent()

{

using (IConnection conn = rabbitMqFactory.CreateConnection())

{

using (IModel channel = conn.CreateModel())

{

//channel.ExchangeDeclare(ExchangeName, "direct", durable: true, autoDelete: false, arguments: null);

channel.QueueDeclare(QueueName, durable: true, autoDelete: false, exclusive: false, arguments: null);

//channel.QueueBind(QueueName, ExchangeName, routingKey: QueueName);

var consumer = new EventingBasicConsumer(channel);

consumer.Received += (model, ea) =>

{

var msgBody = Encoding.UTF8.GetString(ea.Body);

Console.WriteLine(string.Format("***接收时间:{0},消息内容:{1}", DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"), msgBody));

};

channel.BasicConsume(QueueName, noAck: true, consumer: consumer);

  1. //已过时用EventingBasicConsumer代替
  2. //var consumer2 = new QueueingBasicConsumer(channel);
  3. //channel.BasicConsume(QueueName, noAck: true, consumer: consumer);
  4. //var msgResponse = consumer2.Queue.Dequeue(); //blocking
  5. //var msgBody2 = Encoding.UTF8.GetString(msgResponse.Body);
  6. Console.WriteLine("按任意值,退出程序");
  7. Console.ReadKey();
  8. }
  9. }
  10. }

复制代码

  

  但是有些时候,消费者同一时间没有能力处理太多的业务,导致分配过来的队列消息不能及时处理完成,这个时候,我们可以设置BasicQos属性,告诉Broker同一时间将未处理完成的消息分配其他消费者,所以接收消息的地方需要略做修改,代码如下:

复制代码

public static void DirectAcceptExchangeTask()

{

using (IConnection conn = rabbitMqFactory.CreateConnection())

{

using (IModel channel = conn.CreateModel())

{

//channel.ExchangeDeclare(ExchangeName, "direct", durable: true, autoDelete: false, arguments: null);

channel.QueueDeclare(QueueName, durable: true, autoDelete: false, exclusive: false, arguments: null);

channel.BasicQos(prefetchSize: 0, prefetchCount: 1, global: false);//告诉broker同一时间只处理一个消息

//channel.QueueBind(QueueName, ExchangeName, routingKey: QueueName);

var consumer = new EventingBasicConsumer(channel);

consumer.Received += (model, ea) =>

{

var msgBody = Encoding.UTF8.GetString(ea.Body);

Console.WriteLine(string.Format("***接收时间:{0},消息内容:{1}", DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"), msgBody));

int dots = msgBody.Split('.').Length - 1;

System.Threading.Thread.Sleep(dots * 1000);

Console.WriteLine(" [x] Done");

//处理完成,告诉Broker可以服务端可以删除消息,分配新的消息过来

channel.BasicAck(deliveryTag: ea.DeliveryTag, multiple: false);

};

//noAck设置false,告诉broker,发送消息之后,消息暂时不要删除,等消费者处理完成再说

channel.BasicConsume(QueueName, noAck: false, consumer: consumer);

  1. Console.WriteLine("按任意值,退出程序");
  2. Console.ReadKey();
  3. }
  4. }

}

复制代码

  

  <2> RabbitMQ的Topic类型Exchange

  Producter 发送消息代码:

复制代码

///



/// 连接配置

///

private static readonly ConnectionFactory rabbitMqFactory = new ConnectionFactory(){

HostName ="192.168.1.8",UserName="hao",Password="abc123",Port= 5672

};

///



/// 路由名称

///

const string TopExchangeName = "topic.justin.exchange";

  1. //队列名称
  2. const string TopQueueName = "topic.justin.queue";
  3. public static void TopicExchangeSendMsg()
  4. {
  5. using (IConnection conn = rabbitMqFactory.CreateConnection())
  6. {
  7. using (IModel channel = conn.CreateModel())
  8. {
  9. channel.ExchangeDeclare(TopExchangeName, "topic", durable: false, autoDelete: false, arguments: null);
  10. channel.QueueDeclare(TopQueueName, durable: false, autoDelete: false, exclusive: false, arguments: null);
  11. channel.QueueBind(TopQueueName, TopExchangeName, routingKey: TopQueueName);
  12. //var props = channel.CreateBasicProperties();
  13. //props.Persistent = true;
  14. string vadata = Console.ReadLine();
  15. while (vadata != "exit")
  16. {
  17. var msgBody = Encoding.UTF8.GetBytes(vadata);
  18. channel.BasicPublish(exchange: TopExchangeName, routingKey: TopQueueName, basicProperties: null, body: msgBody);
  19. Console.WriteLine(string.Format("***发送时间:{0},发送完成,输入exit退出消息发送", DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss")));
  20. vadata = Console.ReadLine();
  21. }
  22. }
  23. }
  24. }

复制代码

  Customer接收消息代码:

复制代码

///



/// 连接配置

///

private static readonly ConnectionFactory rabbitMqFactory = new ConnectionFactory() {

HostName = "192.168.1.8", UserName = "hao", Password = "abc123", Port = 5672

};

  1. /// <summary>
  2. /// 路由名称
  3. /// </summary>
  4. const string TopExchangeName = "topic.justin.exchange";
  5. //队列名称
  6. const string TopQueueName = "topic.justin.queue";
  7. public static void TopicAcceptExchange()
  8. {
  9. using (IConnection conn = rabbitMqFactory.CreateConnection())
  10. {
  11. using (IModel channel = conn.CreateModel())
  12. {
  13. channel.ExchangeDeclare(TopExchangeName, "topic", durable: false, autoDelete: false, arguments: null);
  14. channel.QueueDeclare(TopQueueName, durable: false, autoDelete: false, exclusive: false, arguments: null);
  15. channel.BasicQos(prefetchSize: 0, prefetchCount: 1, global: false);
  16. channel.QueueBind(TopQueueName, TopExchangeName, routingKey: TopQueueName);
  17. var consumer = new EventingBasicConsumer(channel);
  18. consumer.Received += (model, ea) =>
  19. {
  20. var msgBody = Encoding.UTF8.GetString(ea.Body);
  21. Console.WriteLine(string.Format("***接收时间:{0},消息内容:{1}", DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss"), msgBody));
  22. int dots = msgBody.Split('.').Length - 1;
  23. System.Threading.Thread.Sleep(dots * 1000);
  24. Console.WriteLine(" [x] Done");
  25. channel.BasicAck(deliveryTag: ea.DeliveryTag, multiple: false);
  26. };
  27. channel.BasicConsume(TopQueueName, noAck: false, consumer: consumer);
  28. Console.WriteLine("按任意值,退出程序");
  29. Console.ReadKey();
  30. }
  31. }
  32. }

复制代码

参考资料:

在 Windows 上安装Rabbit MQ 指南(http://www.cnblogs.com/shanyou/p/4067250.html)

.NET 环境中使用RabbitMQ(http://www.cnblogs.com/yangecnu/p/4227535.html)

RabbitMQ Tutorial(http://www.rabbitmq.com/tutorials/tutorial-six-dotnet.html)

源代码下载

知道的越多,不知道的也就越多,多多学习!

NETCore中RabbitMQ的使用的更多相关文章

  1. .NetCore中简单使用EasyNetQ

    前言 我们在.Net中使用RabbitMQ,最原始的就是基于RabbitMQ.Client进行编码,在这个过程中我们需要通过代码约定和维护队列,Exchange等.如果是自行编码封装通用型的Rabbi ...

  2. .NetCore中的日志(2)集成第三方日志工具

    .NetCore中的日志(2)集成第三方日志工具 0x00 在.NetCore的Logging组件中集成NLog 上一篇讨论了.NetCore中日志框架的结构,这一篇讨论一下.NetCore的Logg ...

  3. .NetCore中的日志(1)日志组件解析

    .NetCore中的日志(1)日志组件解析 0x00 问题的产生 日志记录功能在开发中很常用,可以记录程序运行的细节,也可以记录用户的行为.在之前开发时我一般都是用自己写的小工具来记录日志,输出目标包 ...

  4. .NET中RabbitMQ的使用

    概述 MQ全称为Message Queue, 消息队列(MQ)是一种应用程序对应用程序的通信方法.RabbitMQ是一个在AMQP基础上完整的,可复用的企业消息系统.他遵循Mozilla Public ...

  5. AutoMapper在asp.netcore中的使用

    # AutoMapper在asp.netcore中的使用  automapper 是.net 项目中针对模型之间转换映射的一个很好用的工具,不仅提高了开发的效率还使代码更加简洁,当然也是开源的,htt ...

  6. netcore中的缓存介绍

    Cache(缓存)是优化web应用的常用方法,缓存存放在服务端的内存中,被所有用户共享.由于Cache存放在服务器的内存中,所以用户获取缓存资源的速度远比从服务器硬盘中获取快,但是从资源占有的角度考虑 ...

  7. 在netcore中如何注入同一个接口的多个实现

    netcore中自带了Ioc框架,这也影响了我们的编码习惯,以前都是静态类或者直接new对象,现在有了Ioc框架的支持,我们也不必守旧,应当使用起来,接受这种对象管理方式.使用过java的同仁,都习惯 ...

  8. .NetCore中EFCore的使用整理(二)-关联表查询

    EF常用处理关联加载的方式有3中:延迟加载(Lazy Loading).贪婪加载 (Eager Loading)以及显示加载. 一.EF Core  1.1 1.当前的版本,还不支持延迟加载(Lazy ...

  9. .NetCore中EFCore for MySql整理(三)之Pomelo.EntityFrameworkCore.MySql

    一.Pomelo.EntityFrameworkCore.MySql简介 Git源代码地址:https://github.com/PomeloFoundation/Pomelo.EntityFrame ...

随机推荐

  1. MATLAB安装libsvm工具箱的方法

    支持向量机(support vector machine,SVM)是机器学习中一种流行的学习算法,在分类与回归分析中发挥着重要作用.基于SVM算法开发的工具箱有很多种,下面我们要安装的是十分受欢迎的l ...

  2. 《React-Native系列》3、RN与native交互之Callback、Promise

    接着上一篇<React-Native系列>RN与native交互与数据传递,我们接下来研究另外的两种RN与Native交互的机制 一.Callback机制 首先Calllback是异步的, ...

  3. SpringBoot 表单验证

    Valid 注解 JSR 303 校验框架注解类: • @NotNull 注解元素必须是非空 • @Null 注解元素必须是空 • @Digits 验证数字构成是否合法 • @Future 验证是否在 ...

  4. coredump调试小结

    在已经启动的进程中使用gdb,用gdb attach 查看so文件中的函数列表 nm -D *.so 关于c.c++类的gdb调试,强烈推荐一本书:debug hack

  5. MySQL安装配置教程

    环境:Windows 7 旗舰版 64位MySQL版本:mysql-5.5.14-winx64MySQL下载地址:http://dev.mysql.com/downloads/installer/ 1 ...

  6. root run-parts

    crontab的文件格式 分 时 日 月 星期 要运行的命令 第1列分钟0-59 第2列小时0-23(0表示子夜) 第3列日1-31 第4列月1-12 第5列星期0-7(0和7表示星期天) 第6列要运 ...

  7. NumPy数组属性

    NumPy - 数组属性 这一章中,我们会讨论 NumPy 的多种数组属性. ndarray.shape 这一数组属性返回一个包含数组维度的元组,它也可以用于调整数组大小. 示例 1 import n ...

  8. D3.js学习笔记(二)——使用绑定在DOM上的数据

    简单例子 在这个例子中,你将会使用D3.js来将数据绑定到DOM元素上.然后再使用D3.js利用绑定到DOM元素上的数据来更新网页. 在上一章中,我们以下面这个页面作为开始的: <!DOCTYP ...

  9. Html工具类

    import java.io.IOException; import java.io.PrintWriter; import javax.servlet.http.HttpServletRespons ...

  10. iOS唯一标示符引导

         在2013年3月21日苹果已经通知开发者,从2013年5月1日起,访问UIDID的应用将不再能通过审核,替代的方案是开发者应该使用“在iOS 6中介绍的Vendor或Advertising标 ...