上一个教程中,我们创建了一个work queue. 其中的每个task都会被精确的传送到一个worker. 这节,我们将会讲把一个message传送到多个consumers. 这种模式叫做publish/subscribe(发布/订阅).

为了说明这种模式,我们将创建一个简单的日志系统(logging system). 它由两个程序组成,一个是发送日志message并且另一个接收。

最重要的,发布的日志message将会被广播到所有的receivers

Exchangs

前面我们讲的包含下面的:producer,queue,consumer

它的主要思想是producer绝不直接发送任何message到queue. 很多情况下,producer甚至不知道一个message是否会被发送到任何queue.

如图,它会直接发送messages到一个exchange. 而对于exchange,一方面它接收来自producer的message,另一方面它把这些message推送到queues. 至于,messages是否会被发送一个特定的queue或者发送到很多queue或者丢弃,这些规则都由exchange type定义。

Exchange type: direct , topic , headers , fanout.

我们这节主要讲fanout,它会控制广播。

  1. channel.ExchangeDeclare("logs", "fanout");

对于fanout exchange ,它会广播它收到的所有的messages 到它知道的所有的queue.

Listing exchanges

对于列出服务器上的exchanges , 你可以使用rabbitmqctl

  1. sudo rabbitmqctl list_exchanges
The default exchange

在前面的教程中,我们不知道exchanges,但是我们仍然可以发送messages 到queues. 因为我们使用到了一个默认的exchange(a default exchange).这个默认的exchange是被空字符串(“”)定义。

回想下,我们之前怎样发送message

  1. var message = GetMessage(args);
  2. var body = Encoding.UTF8.GetBytes(message);
  3. channel.BasicPublish(exchange: "", //默认的exchange
  4. routingKey: "hello",
  5. basicProperties: null,
  6. body: body);

此时,messages会根据指定的routingKey被路由到queue.

现在,我们可以发布到指定的exchange.

  1. var message = GetMessage(args);var body = Encoding.UTF8.GetBytes(message);
  2. channel.BasicPublish(exchange: "logs",
  3. routingKey: "",
  4. basicProperties: null,
  5. body: body);

Temporary queues

之前我们使用过很多指定名称的queues(例如hello和task_queue). 可以命名一个queue是很重要的,我们可以指定workers到同一个queue。 而且使你可以在多个producers和consumers之前共享这个queue.

We’re also interested only in currently flowing messages not in the old ones. 我们想要最新的message而不是仅仅之前的。

这需要解决两个事情。

  1. 首先,无论什么时候我们连接Rabbit,我们需要一个新的,空的queue。为了达到这个目的,我们可以创建一个带随机名称的queue。更好的办法,我们可以让服务器给我们选择一个随机的queue名称。
  2. 第二,一旦我们断开与consumer的连接,这个queue应该被自动删除。

在.NET客户端中,我们使用下面的语句创建一个带随机名称的queue (when we supply no parameters to QueueDeclare() we create a non-durable, exclusive, autodelete queue with a generated name)

  1. var queueName = channel.QueueDeclare().QueueName;

Bindings

我们已经创建好了exchange和queue,它们之间的关系我们叫做binding. 用来告诉exchange发送messages到queue.

  1. channel.QueueBind(queue: queueName, //绑定
  2. exchange: "logs",
  3. routingKey: "");

现在,在logs exchange上会把messages发到我们的queue。

Listing bindings
  1. rabbitmqctl list_bindings

代码

这种fanout exchanges ,在发送时,会忽视routingKey的值。

EmitLog.cs(发送)

  1. using System;using RabbitMQ.Client;using System.Text;
  2. class EmitLog
  3. {
  4. public static void Main(string[] args)
  5. {
  6. var factory = new ConnectionFactory() { HostName = "localhost" };
  7. using(var connection = factory.CreateConnection())
  8. using(var channel = connection.CreateModel())
  9. {
  10. channel.ExchangeDeclare(exchange: "logs", type: "fanout"); //声明exchange
  11.  
  12. var message = GetMessage(args);
  13. var body = Encoding.UTF8.GetBytes(message);
  14. channel.BasicPublish(exchange: "logs", //发送到logs exchange
  15. routingKey: "",
  16. basicProperties: null,
  17. body: body);
  18. Console.WriteLine(" [x] Sent {0}", message);
  19. }
  20.  
  21. Console.WriteLine(" Press [enter] to exit.");
  22. Console.ReadLine();
  23. }
  24.  
  25. private static string GetMessage(string[] args)
  26. {
  27. return ((args.Length > )
  28. ? string.Join(" ", args)
  29. : "info: Hello World!");
  30. }
  31. }

不允许发送到一个不存在的exchange.

如果没有queue绑定到exchange,messages将会丢失。如果没有consumer正在监听,我们可以安全的丢弃这些message.

ReceiveLogs.cs

  1. using System;using RabbitMQ.Client;using RabbitMQ.Client.Events;using System.Text;
  2. class ReceiveLogs
  3. {
  4. public static void Main()
  5. {
  6. var factory = new ConnectionFactory() { HostName = "localhost" };
  7. using(var connection = factory.CreateConnection())
  8. using(var channel = connection.CreateModel())
  9. {
  10. channel.ExchangeDeclare(exchange: "logs", type: "fanout"); //声明exchange
  11.  
  12. var queueName = channel.QueueDeclare().QueueName; //获得随机queue name
  13. channel.QueueBind(queue: queueName, //定义queue和exchange的关系
  14. exchange: "logs",
  15. routingKey: "");
  16.  
  17. Console.WriteLine(" [*] Waiting for logs.");
  18.  
  19. var consumer = new EventingBasicConsumer(channel); //回调
  20. consumer.Received += (model, ea) =>
  21. {
  22. var body = ea.Body;
  23. var message = Encoding.UTF8.GetString(body);
  24. Console.WriteLine(" [x] {0}", message);
  25. };
  26. channel.BasicConsume(queue: queueName,
  27. autoAck: true,
  28. consumer: consumer);
  29.  
  30. Console.WriteLine(" Press [enter] to exit.");
  31. Console.ReadLine();
  32. }
  33. }
  34. }

参考网址:

https://www.rabbitmq.com/tutorials/tutorial-three-dotnet.html

RabbitMQ学习之Publish/Subscribe(3)的更多相关文章

  1. RabbitMQ入门:发布/订阅(Publish/Subscribe)

    在前面的两篇博客中 RabbitMQ入门:Hello RabbitMQ 代码实例 RabbitMQ入门:工作队列(Work Queue) 遇到的实例都是一个消息只发送给一个消费者(工作者),他们的消息 ...

  2. RabbitMQ学习总结 第四篇:发布/订阅 Publish/Subscribe

    目录 RabbitMQ学习总结 第一篇:理论篇 RabbitMQ学习总结 第二篇:快速入门HelloWorld RabbitMQ学习总结 第三篇:工作队列Work Queue RabbitMQ学习总结 ...

  3. RabbitMQ学习第三记:发布/订阅模式(Publish/Subscribe)

    工作队列模式是直接在生产者与消费者里声明好一个队列,这种情况下消息只会对应同类型的消费者. 举个用户注册的列子:用户在注册完后一般都会发送消息通知用户注册成功(失败).如果在一个系统中,用户注册信息有 ...

  4. 【RabbitMQ】Publish/Subscribe

    Publish/Subscribe 在上一节我们创建了一个work queue.背后的设想为每个任务被分发给明确的消费者.这节内容我们将做一些完全不同的事情 -- 我们将发送一条消息给多个消费者.这种 ...

  5. RabbitMQ(三) -- Publish/Subscribe

    RabbitMQ(三) -- Publish/Subscribe `rabbitmq`支持一对多的模式,一般称为发布/订阅.也就是说,生产者产生一条消息后,`rabbitmq`会把该消息分发给所有的消 ...

  6. (转)RabbitMQ消息队列(四):分发到多Consumer(Publish/Subscribe)

    上篇文章中,我们把每个Message都是deliver到某个Consumer.在这篇文章中,我们将会将同一个Message deliver到多个Consumer中.这个模式也被成为 "pub ...

  7. RabbitMQ消息队列(四):分发到多Consumer(Publish/Subscribe)

    上篇文章中,我们把每个Message都是deliver到某个Consumer.在这篇文章中,我们将会将同一个Message deliver到多个Consumer中.这个模式也被成为 "pub ...

  8. RabbitMQ --- Publish/Subscribe(发布/订阅)

    目录 RabbitMQ --- Hello Mr.Tua RabbitMQ --- Work Queues(工作队列) 前言 在第二篇文章中介绍了 Work Queues(工作队列),它适用于把一个消 ...

  9. RabbitMQ 分发到多Consumer(Publish/Subscribe)

    上篇文章中,我们把每个Message都是deliver到某个Consumer.在这篇文章中,我们将会将同一个Message deliver到多个Consumer中.这个模式也被成为 "pub ...

随机推荐

  1. JMETER 生成测试报告

    JMETER测试报告样例 JMETER 提供的生成测试报告功能,能够生成漂亮的HTML测试报告. 上图是测试统计图 20个用户并发,测试时长一分钟,发起流程320次,没有出错,TPS为6.5,平均发起 ...

  2. Linux命令查找文件目录

    座右铭:长风破浪会有时,直挂云帆济沧海. linux一般查看文件或者目录有几种方法. /查看文件类容--------cat/more/less/head/tail   只能查看文本型(txt) (1) ...

  3. webuploader大文件分片,多线程总结

    项目的新需求是用webuploader来做一个多文件,多线程,并且可以进行分块上传的要求,这些在前面的一篇文章当中足够使用了,但是现在又来一个新的需求,要求上传失败的文件进行重新的上传……心里默默说句 ...

  4. boost与MFC的冲突(new)

    在MFC对话框程序中用boost::signals2时出现了问题, 由于MFC为了方便调试,在debug下重新定义了new #ifdef _DEBUG#define new DEBUG_NEW#end ...

  5. Mybatis-plus中不列出全部字段

    //不列出全部字段 @Test public void test10() { QueryWrapper<User> wrapper = new QueryWrapper<>() ...

  6. c语言中的数据变量类型,大小

    C中有哪些数据类型? 回答: 有两种类型的数据类型,用户定义和预定义.预定义的数据类型是int,char,float,double等,用户使用标签struct,union或enum创建用户定义的数据类 ...

  7. 移动端touch与click区别--移动端开发整理笔记(五)

    移动端用touch还是click? 在移动端开发中,click事件有300ms的延时,由来源于iphone处理双击缩放功能种下的坑.因为用手指在屏幕上快速点击两次,iOS 自带的 Safari 浏览器 ...

  8. Numpy | 23 文件读写

    Numpy 可以读写磁盘上的文本数据或二进制数据. NumPy 为 ndarray 对象引入了一个简单的文件格式:npy. npy 文件用于存储重建 ndarray 所需的数据.图形.dtype 和其 ...

  9. SpringBoot之邮件服务

    springboot 邮件服务 今天在看网上学习微服务的时候顺遍看到了一些关于springboot的文章,写的springboot拓展功能就顺遍学习了一下,接下来给大家分享一下springboot封装 ...

  10. Windows下ActiveMq安装与使用

    一.activeMq安装与启动 Apache Active MQ的官网 :http://activemq.apache.org/ 下载地址: http://activemq.apache.org/ac ...