RabbitMQ系列教程之三:发布\/订阅(Publish\/Subscribe)

在前一个教程中,我们创建了一个工作队列。工作队列背后的假设是每个任务会被交付给一个【工人】。在这一部分我们将做一些完全不同的事情--我们将向多个【消费者】传递信息。这种模式被称为“发布/订阅”。

   为了说明这种模式,我们将构建一个简单的日志系统。它将包括两个程序,第一个将发出日志消息,第二个将接收并打印它们。

   在我们的日志系统中每个接收程序的运行副本都会得到消息。这样我们就可以运行一个接收者程序,将日志记录到磁盘;同时我们可以运行另一个接收者程序,并在屏幕上看到打印出来的日志。

   从本质上讲,已发布的日志消息将被广播到所有的接收者程序。

1、消息交换机【Exchange】

   在教程的前面部分,我们从队列中发送和接收消息。在RabbitMQ中,现在是时候引入全消息模型。

   让我们快速看看我们以前的教程讲了什么:

   【生产者】:就是一个用于发送消息的用户程序
   
   【消费者】:就是一个用于接收和使用消息的用户程序

   【队列】:是一个暂存消息的缓存区

   RabbitMQ消息传递模型的核心思想是,【生产者】不直接发送任何信息到队列。事实上,【生产者】根本就不知道消息是否会被传送到任何队列。

   相反,【生产者】只能发送消息到【消息交换机】。交换是件很简单的事。一方面它接收来自【生产者】的消息,另一方面是将接收到消息推送到队列中。【消息交换机】必须知道它如何处理接收消息的确切方法。是否应该发送到特定队列?它应该被发送到多个队列呢?或者它应该被丢弃。该规则由【消息交换机】的类型来定义。

 这里有一些可用的【消息交换机】的类型:【Direct】直接,【Topic】主题,【Headers】标题和【Fanout】扇出。我们将集中关注最后一个-【Fanout】扇出。让我们创建一个这种类型的【消息交换机】,并给它命名为Logs:

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


   【Fanout】类型的【消息交换机】非常简单。正如你从名字可能猜出的,它只是传播它收到的所有消息去它知道所有的队列中。这正是我们需要我们的日志记录器。

    显示【消息交换机】的列表:

    使用Rabbitmqctl列出在服务器上可以运行的最有用的【消息交换机】

 sudo rabbitmqctl list_exchanges   

    在这个列表中会有一些amq.*【消息交换机】和默认(未命名)消息交换机。这些都是默认创建的,但现在不太可能需要使用它们。

    默认的消息交换机

    在教程前面的部分我们队【消息交换机】是一无所知,但是我依然可以发送消息去想去的队列,那是因为我们使用了默认的【消息交换机】,这些默认的消息 交换机我用使用两个双引号“”来标识。

   我们回忆一下以前是如何发送消息的:

var message = GetMessage(args);var body = Encoding.UTF8.GetBytes(message);
channel.BasicPublish(exchange: "", routingKey: "hello", basicProperties: null, body: body);

   第一个参数是【消息交换机】的名称。空字符串表示默认或未命名的消息交换机:消息会被路由到指定的routingkey名称的队列,如果它存在的话。

   现在,我们可以发布到我们命名的【消息交换机】:

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


2、临时队列

   也许你还记得以前我们使用的队列所指定的名称(记得Hello和task_queue吗?)对我们来说,能够给一个队列指定名称是至关重要的--因为我们需要把【Worker】指向同一个队列。如果要在【生产者】和【消费者】之间共享队列,给队列命名是很重要的。

    但这不是我们的日志记录器的情况。我们想听到所有的日志消息,而不仅仅是其中的一个子集。我们也只对当前刚刚收到的消息感兴趣,而不是对旧的。为了解决上述问题,我们需要做两件事。

   首先,无论何时当我们连接到Rabbit的时候,我们都需要一个新的并且是空的队列。要做到这一点,我们可以创建一个具有随机名称的队列,或者,甚至更好一点-让服务器为我们选择一个随机队列名称。

   其次,一旦我们断开与【消费者】的队列就应该自动删除该队列。

   在.NET客户端中,当我们没有为queueDeclare()提供参数时,我们创建了一个具有生成名称的非持久,排他,自动删除队列:

var queueName = channel.QueueDeclare().QueueName;

   在这点上,QueueName包含随机队列名称。例如,它可能看起来像amq.gen-jzty20brgko-hjmujj0wlg。

3、绑定【Binding】 

frameborder="0" scrolling="no" style="">

 我们已经创建了一个【Fanout】类型的【消息交换机】和队列。现在我们需要告诉【消息交换机】向我们的队列发送消息。【消息交换机】和【队列】之间的关系称为绑定。


channel.QueueBind(queue: queueName, exchange: "logs", routingKey: "");


   从现在开始,日志的【消息交换机】就可以将消息推送到我们定义的队列中去了。

   我们可以通过以下语句查看【binding】列表数据:
  

rabbitmqctl list_bindings

4、把所有的代码整合到一起

   【生产者】的程序,它发出的日志消息,看起来并没有和以前的教程有很大的不同。最重要的变化是,我们现在想发送的消息是到达我们指定名称的日志【消息交换机】,而不是无名的。我们在发送消息的时候需要提供一个routingkey表示的名称,但【Fanout】类型的【消息交换机】会容忽视该routingKey的值的。这里有EmitLog.cs文件代码:

相关文章:

  • RabbitMQ系列教程之一:我们从最简单的事情开始!Hello World

  • RabbitMQ系列教程之二:工作队列(Work Queues)

  • 如何优雅的使用RabbitMQ

  • .NET 使用 RabbitMQ 图文简介

  • RabbitMQ 高可用集群搭建及电商平台使用经验总结

  • 搭建高可用的rabbitmq集群 + Mirror Queue + 使用C#驱动连接

  • RabbitMQ消息队列应用

  • 体验Rabbitmq强大的【优先级队列】之轻松面对现实业务场景

原文地址:http://www.cnblogs.com/PatrickLiu/p/6943830.html


.NET社区新闻,深度好文,微信中搜索dotNET跨平台或扫描二维码关注

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/324885.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

路径使用场景

路径的使用场景 路径适用的范围:jsp页面中link script img a form 等 如果访问的是其他服务器中的项目文件只可以使用绝对路径 如果访问的是同一个服务器中的其他项目推荐使用根路径 如果访问的是同一个服务器中的同一个项目中的文件推荐使用相对base标签的路径 修改…

《此生未完成》痛句摘抄(3)

System.out.print("今天开始继续读书摘录"); //不知道官方让不让我在博客里面记录 //如果不让的话我可能得转到别的上面记录 System.out.print("现在开始看《此生未完成》"); System.out.println("今天是第三天!");有时候常常会想到那…

【jzoj】2018.1.30NOIP普及组——模拟赛D组

_ 前言 写博客时间 2018/1/30 22:36。 感想:出数据的dalao我服! 正题 题目1:二项式展开式(jzoj2254) 输入一个整数,求展开(ab)^n。展开方式为 (ab)^n?a^n?a^(n-1)b?a^(n-2)b^2…?b^n 其中” ? “…

开发人员也要懂点的测试知识

转载自 开发人员也要懂点的测试知识 本文来自于作者投稿,作者陈彩华,贝聊后端开发工程师。 最近参加了保利威测试总监李乐的《互联网测试姿势》为主题的分享交流会,收获颇丰,作为一个开放,秉承“不懂产品和测试的开…

Java8中接口的新特性

知识点1: 接口中定义的静态方法,只能通过接口来调用。 CompareA.method1(); 知识点2: 通过实现类的对象,可以调用接口中的默认方法。如果实现类重写了接口中的默认方法,调用时,仍然调用的是重写以后的方…

转发和重定向

需求说明 用户输入并提交登录信息 登录成功,跳转到success.jsp页面 登录失败,跳转回login.jsp页面 理解图 区别详解

实现自己的.NET Core配置Provider之Yaml

YAML是一种更适合人阅读的文件格式,很多大型的项目像Ruby on Rails都选择YAML作为配置文件的格式。如果项目的配置很少,用JSON或YAML没有多大差别。看看rails项目中的配置文件,如果用JSON写试试什么感受吧。 在《实现自己的.NET Core配置Pro…

信息时代与人工智能时代的教育变革

求关注求转发本文是雄雄的小课堂投稿的第 1 篇文章,作者:小溪时代发展迅速,学校的教育体系却没有很大的改变。比如现在的学生们还是学着几百年前的东西。试想一下,这是否真的适合我们这个时代,亦或真的需要改变了。教育…

【jzoj】2018.1.31 NOIP普及组——D组模拟赛

前言 今天题目比较水and我进了C组,不过太太太太太太太太太太太太太太太太绝望了QAQ。所以我也没有做C组的题。写完博客我就做O(∩_∩)O。 正题 题1:奇数统计(jzoj1547) 就是输入n个数,输出出现次数为奇数的一个数&a…

请求和响应向更多内容

1、请求更多方法 /*******[1]获得req对象中的头信息*(了解)*******************/Enumeration<String> headerNames req.getHeaderNames();while(headerNames.hasMoreElements()){String next headerNames.nextElement();System.out.println(next"-----"req.…

大数据Big Data

转载自 大数据Big Data 2012年本站曾对大数据预测&#xff1a;如果说2012年是大数据概念为人所知、引人瞩目、小试牛刀的一年&#xff0c;那么2013年大数据将会实现产品部署&#xff0c;早期投资获得回报&#xff0c;一小部分的产业被颠覆。到了2014年&#xff0c;各种大数据项…

RabbitMQ系列教程之四:路由(Routing)

在上一个教程中&#xff0c;我们构建了一个简单的日志系统&#xff0c;我们能够向许多消息接受者广播发送日志消息。在本教程中&#xff0c;我们将为其添加一项功能 &#xff0c;这个功能是我们将只订阅消息的一个子集成为可能。 例如&#xff0c;我们可以只将关键的错误消息输…

《此生未完成》痛句摘抄(4)

System.out.print("今天开始继续读书摘录"); //不知道官方让不让我在博客里面记录 //如果不让的话我可能得转到别的上面记录 System.out.print("现在开始看《此生未完成》"); System.out.println("今天是第四天&#xff01;"); System.out.print…

【jzoj3734,Usaco2014Open银组】双导航(gpsdual)

前言 这是今天C组的题&#xff0c;闲得无聊做了一会&#xff0c;结果就对了233。这算是学了SPFA之后的第一次实战了。反正其他C组题我也不想做了。好了现在bi~~&#xff08;系统自动屏蔽&#xff09;也在做这道题。 还有这道题的名字叫 正题 题目 一个有向图&#xff0c;有…

Java代码到底是如何编译成机器指令的。

转载自 Java代码到底是如何编译成机器指令的。 在《Java代码的编译与反编译》中&#xff0c;有过关于Java语言的编译和反编译的介绍。我们可以通过javac命令将Java程序的源代码编译成Java字节码&#xff0c;即我们常说的class文件。这是我们通常意义上理解的编译。 但是&am…

DDD理论学习系列(3)-- 限界上下文

1. 引言 限界上下文可以拆分为两个词&#xff0c;限界和上下文。限界&#xff1a;是指一个界限&#xff0c;具体的某一个范围。上下文&#xff1a;个人理解就是语境。 比如我们常说的段子&#xff1a; “我想静静。”这个句子一般是想表达“我想静一静”的意思。但是我们却把…

Git GitHub Gitee GitLab

Git Git 常用命令 git --version 查看 git 版本 git config --global user.name 用户名 设置用户签名 git config --global user.email 邮箱 设置用户签名 git init 初始化本地库 dellDESKTOP-VSDN0Q2 MINGW64 /d/Git-Space/SH0720 $ git init Initialized empty Git reposi…

《下辈子还教书》经典语录(1)

System.out.print("今天开始继续读书摘录"); //不知道官方让不让我在博客里面记录 //如果不让的话我可能得转到别的上面记录 System.out.print("现在开始看《下辈子还教书》"); System.out.println("今天是第1天&#xff01;");自从语文由文化的…

jzoj1478-堆排序【堆】

题目 一些数&#xff0c;用堆把它们从小到大排好 解题思路 每个堆的开头是最大&#xff08;小&#xff09;的。每次把开头读取出来&#xff0c;然后把a[num]提取上来&#xff0c;然后num减1在把新的a[1]降到合适的位置。 代码 #include<cstdio> using namespace std…

异常体系结构

常见异常 异常体系结构 java.lang.Throwable |-----java.lang.Error:一般不编写针对性的代码进行处理 |-----java.lang.Exception:可以进行异常的处理 |------编译时异常(checked) |-----IOException |-----FileNotFoundException |-----ClassNotFoundException |------运行时…