RabbitMQ 的行为艺术

  好像,今天已经是 2 月 28 号了。

  听说,29、30、31 号放假。

  据说,有图,有真相。

目录

  • 简介

  • 环境搭建

  • 示例一:简单的 Hello World

  • 示例二:发布/订阅模式

  • 尝试发现 - 新物种 EasyNetQ

简介

  RabbitMQ:一个消息系统,基于 AMQP 系统协议,由 Erlang 语言开发。

  优点:健壮、使用简单、开源和支持各种流行的语言(如 Python、java、.NET)等。

  MQ(Message Queue):消息队列的简称,是一种应用程序之间的通信机制。

  作用:将部分无需立即回调获取结果,并且耗时的操作,使用异步处理的方式提高服务器的吞吐量及性能。如:日志记录。

图:简单的通信方式,及加入 MQ 后的变化

 
     A 端:生产者将消息写(插)入队列;
     MQ(队列) :中间件,消息的载体;
     B 端:消费者从队列读(取)出消息。
 
  MQ 特点:消费者 - 生产者模型的一种表现形式。
 

环境搭建

  1.官网下载安装包:http://www.rabbitmq.com/ ;

  2.安装时会提示你下载 Erlang 语言环境;

  3.启动安装完的服务:RabbitMQ;

  4.在 cmd 中指向 sbin 目录,并输入以下命令,才能打开 WEB 管理界面:

rabbitmq-plugins enable rabbitmq_management

  5.默认 url:http://localhost:15672/#/

示例一:简单的 Hello World

  P(Producer):生产者,意味着发送;

  Queue:队列,本质上是一个无限的缓冲区,可以储存尽可能多的信息;

  C(Consumer):消费者,等待并接收消息。

  【备注】生产者和消费者不需要驻留在同一台服务器上。

  Producer.cs

     public class Producer  
{
public static void Send()
{
var factory = new ConnectionFactory { HostName = "localhost" }; //创建连接对象,基于 Socket
using (var connection = factory.CreateConnection())
{
//创建新的渠道、会话
using (var channel = connection.CreateModel())
{
//声明队列
channel.QueueDeclare(queue: "hello", //队列名
durable: false, //持久性
exclusive: false, //排他性
autoDelete: false, //自动删除
arguments: null); const string message = "Hello World!";
var body = Encoding.UTF8.GetBytes(message); channel.BasicPublish(exchange: "", //交换机名
routingKey: "hello", //路由键
basicProperties: null,
body: body);
}
}
}
}

  【备注】队列名如果已存在,将不会重复创建。假设队列已存在,修改 channel.QueueDeclare() 方法内的参数后启动会出现异常。

  【备注】消息内容是一个字节数组。

 
  Consumer.cs
     class Consumer
{
public static void Receive()
{
var factory = new ConnectionFactory() { HostName = "localhost" }; using (var connection = factory.CreateConnection())
{
using (var channel = connection.CreateModel())
{
channel.QueueDeclare(queue: "hello",
durable: false,
exclusive: false,
autoDelete: false,
arguments: null); //创建基于该队列的消费者,绑定事件
var consumer = new EventingBasicConsumer(channel);
consumer.Received += (model, ea) =>
{
var body = ea.Body; //消息主体
var message = Encoding.UTF8.GetString(body);
Console.WriteLine(" [x] Received {0}", message);
}; //启动消费者
channel.BasicConsume(queue: "hello", //队列名
noAck: true, //false:手动应答;true:自动应答
consumer: consumer); Console.Read();
}
}
}
}

  【疑问】在消费者的类里面为什么会再次声明队列(channel.QueueDeclare())呢?-- 因为接收方可能会在发送方启动前启动,这是出于保险起见。

示例二:发布/订阅模式

 
 

  1.Exchange 交换机和 Exchange Type 交换类型  

  RabbitMQ 消息传递模型的核心思想是,生产者不会直接将消息发给队列。

  这里我们将引入新的名词 Exchange(交换机)。交换机传递消息的类型也有很多种:direct, topic, headers(不常用) 和 fanout,我们称之为交换类型。

图:Direct

图:Fanout

图:Topic

--上述 3 张图来源:http://m.blog.csdn.net/article/details?id=52262850

  

  这里,创建一个名为 “logs” 的交换机,它的类型为广播类型(fanout:可以将收到的所有消息,广播给所有已知的队列)。

channel.ExchangeDeclare(exchange: "logs",   //交换机名
type: "fanout"); //交换类型

  

  2.临时队列

  作为消费者,我们有时候只需要一些新的(或者空的)队列,此时,更好的方式就是让它自动生成一个随机名字的队列;其次,当队列连接中断时会选择自动删除对应的消费者。

  创建一个非持久,有排他性和自动删除特性的队列(无参时)。

var queueName = channel.QueueDeclare().QueueName;

  3.Binding 绑定

  【疑问】有了 Exchange 和 channel,这时,还需要什么东西呢?-- 我们要创建 Exchange 和 channel 关系的桥梁,这个桥梁称之为 Binding(绑定)。

channel.QueueBind(queue: queueName,
exchange: "logs",
routingKey: "");
 
     class Producer
{
public static void Send()
{
var factory = new ConnectionFactory()
{
HostName = "localhost",
Port = ,
UserName = "guest",
Password = "guest"
}; using (var connection = factory.CreateConnection())
{
using (var channel = connection.CreateModel())
{
channel.ExchangeDeclare(exchange: "logs", //交换机名
type: "fanout"); //交换类型 // Guid
var message = Guid.NewGuid().ToString();
var body = Encoding.UTF8.GetBytes(message);
channel.BasicPublish(exchange: "logs",
routingKey: "",
basicProperties: null,
body: body); Console.WriteLine(" [x] Sent {0}", message);
} Console.WriteLine(" Press [enter] to exit.");
Console.ReadLine();
}
}
}

Producer.cs //生产者

     class Reciver
{
public static void Recive()
{
var factory = new ConnectionFactory()
{
HostName = "localhost",
Port = ,
UserName = "guest",
Password = "guest"
}; using (var connection = factory.CreateConnection())
using (var channel = connection.CreateModel())
{
channel.ExchangeDeclare(exchange: "wen_logs", //交换机名
type: "fanout"); //交换类型 //创建队列
var queueName = channel.QueueDeclare().QueueName;
channel.QueueBind(queue: queueName,
exchange: "wen_logs",
routingKey: ""); Console.WriteLine(" [*] Waiting for logs."); var consumer = new EventingBasicConsumer(channel);
consumer.Received += (model, ea) =>
{
var body = ea.Body;
var message = Encoding.UTF8.GetString(body);
Console.WriteLine(" [x] {0}", message);
};
channel.BasicConsume(queue: queueName,
noAck: true,
consumer: consumer); Console.WriteLine(" Press [enter] to exit.");
Console.ReadLine();
}
}
}

Reciver.cs //接收者

尝试发现 - 新物种 EasyNetQ

  这都不是事!EasyNetQ,看名字就知道,搞定 MQ,So easy!

  连接 RabbitMQ 代理:

var bus = RabbitHutch.CreateBus("host=localhost");

  发布:

bus.Publish(message);

  订阅:

bus.Subscribe<MyMessage>("my_subscription_id", msg => Console.WriteLine(msg.Text));

  下面我们通过 Demo 来感受一下 Easy 的程度吧,创建项目(效果图如下,附 Demo 下载):

  Wen.EasyNetQDemo.Model:类库

  Wen.EasyNetQDemo.Publisher,Wen.EasyNetQDemo.Subscriber:控制台应用程序,都使用 Nuget 直接安装 EasyNetQ 包,都引用类库 Model。

 
  Demo.cs

    public class Demo
{
public string Message { get; set; }
}

  Publisher

 using System;
using EasyNetQ;
using Wen.EasyNetQDemo.Model; namespace Wen.EasyNetQDemo.Publisher
{
internal class Program
{
private static void Main(string[] args)
{
using (var bus = RabbitHutch.CreateBus("host=localhost"))
{
string input;
Console.WriteLine("请输入信息。 如果是“esc” 将退出当前窗口。"); while ((input = Console.ReadLine()) != "esc")
{
bus.Publish(new Demo
{
Message = input
});
} }
}
}
}

  【备注】RabbitHutch.CreateBus() 方法可以创建一个简单的发布/订阅和包含请求/响应 API 的消息总线。

  Subscriber

 using System;
using EasyNetQ;
using Wen.EasyNetQDemo.Model; namespace Wen.EasyNetQDemo.Subscriber
{
internal class Program
{
private static void Main(string[] args)
{
using (var bus = RabbitHutch.CreateBus("host=localhost"))
{
bus.Subscribe<Demo>("test", HandleDemo); Console.WriteLine("监听信息中...输入“return”将退出当前窗口!");
Console.ReadLine();
}
} private static void HandleDemo(Demo demo)
{
Console.ForegroundColor = ConsoleColor.Green;
Console.WriteLine($"Got message: {demo.Message}");
Console.ResetColor();
}
}
}

图:效果图

「世事洞明皆学问 人情练达即文章」

【博主】反骨仔

[.NET] RabbitMQ 的行为艺术的更多相关文章

  1. RabbitMQ 的行为艺术

    RabbitMQ 的行为艺术 目录 简介 环境搭建 示例一:简单的 Hello World 示例二:发布/订阅模式 尝试发现 - 新物种 EasyNetQ 简介 RabbitMQ:一个消息系统,基于 ...

  2. Redis 小白指南(三)- 事务、过期、消息通知、管道和优化内存空间

    Redis 小白指南(三)- 事务.过期.消息通知.管道和优化内存空间 简介 <Redis 小白指南(一)- 简介.安装.GUI 和 C# 驱动介绍> 讲的是 Redis 的介绍,以及如何 ...

  3. 【框架学习与探究之消息队列--EasyNetQ(1)】

    前言 本文欢迎转载,实属原创,本文原始链接地址:http://www.cnblogs.com/DjlNet/p/7603554.html 废话 既然都是废话了,所以大家就可以跳过了,这里是博主有事没事 ...

  4. Open Source

    资源来源于http://www.cnblogs.com/Leo_wl/category/246424.html RabbitMQ 安装与使用 摘要: RabbitMQ 安装与使用 前言 吃多了拉就是队 ...

  5. 消息队列——RabbitMQ学习笔记

    消息队列--RabbitMQ学习笔记 1. 写在前面 昨天简单学习了一个消息队列项目--RabbitMQ,今天趁热打铁,将学到的东西记录下来. 学习的资料主要是官网给出的6个基本的消息发送/接收模型, ...

  6. 读书笔记:JavaScript DOM 编程艺术(第二版)

    读完还是能学到很多的基础知识,这里记录下,方便回顾与及时查阅. 内容也有自己的一些补充. JavaScript DOM 编程艺术(第二版) 1.JavaScript简史 JavaScript由Nets ...

  7. RabbitMq应用二

    在应用一中,基本的消息队列使用已经完成了,在实际项目中,一定会出现各种各样的需求和问题,rabbitmq内置的很多强大机制和功能会帮助我们解决很多的问题,下面就一个一个的一起学习一下. 消息响应机制 ...

  8. 如何优雅的使用RabbitMQ

    RabbitMQ无疑是目前最流行的消息队列之一,对各种语言环境的支持也很丰富,作为一个.NET developer有必要学习和了解这一工具.消息队列的使用场景大概有3种: 1.系统集成,分布式系统的设 ...

  9. 有趣的 CSS 像素艺术

    原文地址:https://css-tricks.com/fun-times-css-pixel-art/#article-header-id-4 译者:nzbin 友情提示:由于国内网络的原因,Cod ...

随机推荐

  1. UVa 10911 - Forming Quiz Teams

    题目大意:给出2*n个点,将这些点配成n对,使得所有点对中两点的距离之和尽量小. 用一个整数的二进制形式表示一个集合的子集,以每个集合为状态进行状态转移.具体参见<算法竞赛入门经典>9.5 ...

  2. linux环境下Vim的配置

    原文链接:http://blog.chinaunix.net/uid-26826958-id-3272375.html  (本文转自此链接中的部分内容,但做了适当修改) 安装vim命令:sudo ap ...

  3. 升级wamp5集成安装包 php5.2到php5.3

    平时xp下面都使用wamp5集成开发 但php的空间命名需要php5.3 才支持,而且公司系统大部分都使用5.3,很多函数与5.2是不同的 难的在xp下面手动安装,集成包使用很方便,配置,快捷键都很不 ...

  4. Memo 的当前行、当前列与当前字符

    procedure TForm1.Memo1Click(Sender: TObject); begin   Text := Format('当前列:%d, 当前行:%d', [Memo1.CaretP ...

  5. #最小生成树# #kruskal# ----- OpenJudge丛林中的路

    最小生成树 一个有 n 个结点的连通图的生成树是原图的极小连通子图,且包含原图中的所有 n 个结点,并且有保持图连通的最少的边.最小生成树可以用kruskal(克鲁斯卡尔)算法或prim(普里姆)算法 ...

  6. python访问sqlserver

    #coding=utf-8 #!/usr/bin/env python#---------------------------------------------------------------- ...

  7. HDU5878(打表)

    I Count Two Three Time Limit: 3000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others ...

  8. venom结合Metasploit绕过360安全卫士

    原理:msfvenom是msfpayload和msfencode的结合体,利用msfvenom生成shellcode,venom生成工具使用了 一些 Veil-Evasion.py, unicorn. ...

  9. CSS3知识点整理(一)----基本样式

    (一) 在编写CSS3样式时,不同的浏览器可能需要不同的前缀.它表示该CSS属性或规则尚未成为W3C标准的一部分,是浏览器的私有属性,虽然目前较新版本的浏览器都是不需要前缀的,但为了更好的向前兼容前缀 ...

  10. [Android ADB] An auto-input method for Android and Windows

    The Valentine's Day is coming. Here is an auto-input method and you may use it to send multiple word ...