一、Publish/Subscribe(发布/订阅)(using the Java Client)

	在前面的教程中,我们创建了一个work Queue(工作队列)。工作队列背后的假设是每个任务是交付给一个工作者(worker) 也就是均匀分给每个消费者。在本部分,我们将做一些完全不同的事情,我们将提供一个消息到多个消费者。这种模式被称为“发布/订阅”。
	为了说明这个模式,我们将构建一个简单的日志系统。它将包括两个项目:
  1. 第一个将发出日志消息
  2. 第二个将接收并打印它们。
	在我们的日志系统,每运行一次,接收器项目将得到消息的副本。这样我们能够运行一个接收机并且可以直接记录到磁盘,同时我们可以运行另一个接收器,看到屏幕上的日志。
注:从本质上讲,发表日志消息广播给所有的接收者。
	下面让我们脑中带几个问题,让我们一步一步去解决:
  • 如果我把消息分配给所有的消费者,我们将怎么做呢?

二、Exchanges(交换机)

在前部分的教程中,我们从一个队列发送和接收消息。现在是时候让Rabbit推出完整的消息模型。
让我们快速复习我们前面的教程::
  • 生产者是一个用户发送消息的应用程序。
  • 一个队列是存储消息的缓冲区。
  • 消费者是一个用户应用程序接收消息。

RabbitMQ的消息模型的核心思想是,生产者从未直接向队列发送任何消息。实际上,经常生产者甚至不知道消息是否会被运送到任何队列。

相反,生产者只能发送Exchanges(消息交换区)。交换是一个非常简单的事情。一方面它从生产者那收到消息并推他们到另一边队列。交换区必须知道如何处理它收到一条消息:

  1. 它应该被加到一个特定的队列吗?
  2. 它应该被加到多队列?
  3. 或者它应该丢弃吗?
交换的规则定义的类型。
如上图所示:X表示Exchange(交换机);
有一些可用的交换类型direct, topic, headers and fanout。我们将专注于最后一个——fanout。让我们创建一个这种类型的交换,称之为日志:
channel.exchangeDeclare("logs", "fanout");
fanout交换非常简单。你大概可以猜到的名字,只是广播所有的消息接收队列它知道。而这正是我们需要为我们的记录器。
问题:
① exchange list 列出所有 (交换机)列表
$ sudo rabbitmqctl list_exchanges
Listing exchanges ...
direct
amq.direct direct
amq.fanout fanout
amq.headers headers
amq.match headers
amq.rabbitmq.log topic
amq.rabbitmq.trace topic
amq.topic topic
logs fanout
...done.
在此列表中有一些amq* 交换器 与默认(匿名)交换。这些都是默认创建的,但可能你不需要使用它们。
② 缺省名字的 exchange(交换机)
在前部分的教程中我们对exchange 一无所知,,但仍然能够将消息发送到队列。这是可能的,因为我们是使用一个默认的交换,我们确定的空字符串(" ")

记得之前我们发布一个消息:
    channel.basicPublish("", "hello", null, message.getBytes());
第一个参数是该交换区的名称;空字符串表示默认或无名的交换,:如果routingKey存在的话,消息路由到指定的队列的名称。
现在,我们可以发布我们的交换器:
    channel.basicPublish( "logs", "", null, message.getBytes());

三、Temporary queues(临时队列)

	你可能记得以前我们使用的队列都是指定名称的(还记得hello和task_queue吗?)。对我们来说命名一个队列是至关重要的,
	当你想在生产者和消费者中分享队列的时候,给一个队列的名称是必须的。	
    但是那些都不是日志记录系统所需要的,我们希望能够获得所有的日志信息,而不只是其中的一部分,而且我们只对当前正在传递的信息感兴趣,
对旧的日志信息不感兴趣,要解决这些问题,我们需要分两个步骤:
  • 首先当我们链接到RabbitMQ服务器的时候,需要一个新的、空的队列,为了做到这点,可以创建一个随机名的队列,
或者更好的方法就是让服务器选择一个随机的队列名。
  • 其次,当断开与队列的连接时,消费者应该被自动删除掉
在Java客户端,我们通过一个无参数的queueDeclare()方法为我们创建一个非持久的、唯一的、能自动删除的队列与队列名称
 String queueName = channel.queueDeclare().getQueue();

在这点上,queueName包含了一个随机队列名称。例如它可能看起来像amq.gen-JzTY20BRgKO-HjmUJj0wLg。

四、Bindings(绑定)

我们已经创建了一个fanout exchange和一个队列,现在我们需要告诉exchange去发送消息到队列中,exchange和队列之间的关系被称为一个绑定(binding)

	channel.queueBind(queueName, "logs", "");

注意:从现在开始我们从logs exchange将被添加消息到队列中,使用rabbitmqctl list_bingdins能列出所有的绑定。



五、Putting it all together(发布者/订阅者 实现)

生产者代码和之前的发送消息的代码并没有太大的区别,最重要的变化是,我们现在要将发布的消息传递给logs exchange来代替无名的exchange(之前的是"")
在发送消息时需要提供一个routingKey,它对于fanout exchange是非常重要的,不能被忽视的,这里的EmitLog.java代码如下
  1. </pre><pre name="code" class="java">import java.io.IOException;
  2. import com.rabbitmq.client.ConnectionFactory;
  3. import com.rabbitmq.client.Connection;
  4. import com.rabbitmq.client.Channel;
  5. public class EmitLog {
  6. private static final String EXCHANGE_NAME = "logs";
  7. public static void main(String[] argv)
  8. throws java.io.IOException {
  9. ConnectionFactory factory = new ConnectionFactory();
  10. factory.setHost("localhost");
  11. Connection connection = factory.newConnection();
  12. Channel channel = connection.createChannel();
  13. channel.exchangeDeclare(EXCHANGE_NAME, "fanout");
  14. String message = getMessage(argv);
  15. channel.basicPublish(EXCHANGE_NAME, "", null, message.getBytes());
  16. System.out.println(" [x] Sent '" + message + "'");
  17. channel.close();
  18. connection.close();
  19. }
  20. //...
  21. }

接收端:

  1. import java.io.IOException;
  2. import com.rabbitmq.client.ConnectionFactory;
  3. import com.rabbitmq.client.Connection;
  4. import com.rabbitmq.client.Channel;
  5. import com.rabbitmq.client.QueueingConsumer;
  6. public class ReceiveLogs {
  7. private static final String EXCHANGE_NAME = "logs";
  8. public static void main(String[] argv)
  9. throws java.io.IOException,
  10. java.lang.InterruptedException {
  11. ConnectionFactory factory = new ConnectionFactory();
  12. factory.setHost("localhost");
  13. Connection connection = factory.newConnection();
  14. Channel channel = connection.createChannel();
  15. channel.exchangeDeclare(EXCHANGE_NAME, "fanout");
  16. String queueName = channel.queueDeclare().getQueue();
  17. channel.queueBind(queueName, EXCHANGE_NAME, "");
  18. System.out.println(" [*] Waiting for messages. To exit press CTRL+C");
  19. QueueingConsumer consumer = new QueueingConsumer(channel);
  20. channel.basicConsume(queueName, true, consumer);
  21. while (true) {
  22. QueueingConsumer.Delivery delivery = consumer.nextDelivery();
  23. String message = new String(delivery.getBody());
  24. System.out.println(" [x] Received '" + message + "'");
  25. }
  26. }
  27. }

像以前一样,我们开始做编译

$ javac -cp rabbitmq-client.jar EmitLog.java ReceiveLogs.java

如果你想将日志保存到一个文件,打开一个控制台并运行

$ java -cp .:commons-io-1.2.jar:commons-cli-1.1.jar:rabbitmq-client.jar ReceiveLogs > logs_from_rabbit.log

如果你想看到日志在你的屏幕上,产生一个新的终端并运行:

$ java -cp .:commons-io-1.2.jar:commons-cli-1.1.jar:rabbitmq-client.jar ReceiveLogs

发布日志类型:

$ java -cp .:commons-io-1.2.jar:commons-cli-1.1.jar:rabbitmq-client.jar EmitLog

使用rabbitmqctl list_bindings实际上您可以验证绑定和队列的代码是否是我们想要的? 有两个ReceiveLogs。

$ sudo rabbitmqctl list_bindings
Listing bindings ...
logs exchange amq.gen-JzTY20BRgKO-HjmUJj0wLg queue []
logs exchange amq.gen-vso0PVvyiRIL2WoV3i48Yg queue []
...done.

RabbitMQ学习总结(5)——发布和订阅实例详解的更多相关文章

  1. 入门学习Linux常用必会命令实例详解

    Linux提供了大量的命令,利用它可以有效地完成大量的工作,如磁盘操作.文件存取.目录操作.进程管理.文件权限设定等.所以,在Linux系统上工作离不开使用系统提供的命令.要想真正理解Linux系统, ...

  2. Part1.2 、RabbitMQ -- Publish/Subscribe 【发布和订阅】

    python 目录 (一).交换 (Exchanges) -- 1.1 武sir 经典 Exchanges 案例展示. (二).临时队列( Temporary queues ) (三).绑定(Bind ...

  3. expect学习笔记及实例详解【转】

    1. expect是基于tcl演变而来的,所以很多语法和tcl类似,基本的语法如下所示:1.1 首行加上/usr/bin/expect1.2 spawn: 后面加上需要执行的shell命令,比如说sp ...

  4. 官网实例详解-目录和实例简介-keras学习笔记四

    官网实例详解-目录和实例简介-keras学习笔记四 2018-06-11 10:36:18 wyx100 阅读数 4193更多 分类专栏: 人工智能 python 深度学习 keras   版权声明: ...

  5. JavaScript学习笔记-实例详解-类(二)

    实例详解-类(二)   //===给Object.prototype添加只读\不可枚举\不可配置的属性objectId(function(){ Object.defineProperty(Object ...

  6. JavaScript学习笔记-实例详解-类(一)

    实例详解-类(一): //每个javascript函数(除了bind())都自动拥有一个prototype对象// 在未添加属性或重写prototype对象之前,它只包含唯一一个不可枚举属性const ...

  7. 这个贴子的内容值得好好学习--实例详解Django的 select_related 和 prefetch_related 函数对 QuerySet 查询的优化

    感觉要DJANGO用得好,ORM必须要学好,不管理是内置的,还是第三方的ORM. 最最后还是要到SQL.....:( 这一关,慢慢练啦.. 实例详解Django的 select_related 和 p ...

  8. Cocos2d-x 3.X手游开发实例详解

    Cocos2d-x 3.X手游开发实例详解(最新最简Cocos2d-x手机游戏开发学习方法,以热门游戏2048.卡牌为例,完整再现手游的开发过程,实例丰富,代码完备,Cocos2d-x作者之一林顺和泰 ...

  9. 「微信小程序」PHP异步进程async-helper实例详解

    PHP异步进程async-helper实例详解 PHP 的异步进程助手,借助于 AMQP 实现异步执行 PHP 的方法,将一些很耗时.追求高可用.需要重试机制的操作放到异步进程中去执行,将你的 HTT ...

随机推荐

  1. FreeMarker 语法 list

    一.java 代码 @Test public void testFreeMarker() throws Exception { //1.创建一个模板文件 //2.创建一个Configuration对象 ...

  2. 使用githug游戏提高git水平

  3. 关于excel导出

    转载自:https://blog.csdn.net/ljj_9/article/details/50395688 //一个excel表格: HSSFWorkbook wb = new HSSFWork ...

  4. Guava工具类

    原文链接:http://blog.csdn.net/mnmlist/article/details/53425865 Objects类 Objects类有几个比较不错的方法,toString.hash ...

  5. iframe显示滚动栏

    子页面通过iframe载入.出现了竖向滚动栏 最后查出原因:文档申明 iframe有滚动栏的页面的文档申明 <!DOCTYPE html> 改成例如以下即可了 <!DOCTYPE H ...

  6. poj 1182 食物链 &amp;&amp; nyoj 207(种类并查集)

    食物链 Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 52414   Accepted: 15346 Description ...

  7. BZOJ4031——HEOI小z的房间

    题意:求某网格图生成树个数,对1e9取模 题解:题目是裸的Matrix-Tree定理,这不是我要说的重点,重点是对于这个取模的处理. 由于这不是个质数,所以不能直接乘逆元来当除法用.直接高斯消元肯定是 ...

  8. 原来C++之父在大摩工作呀,并且还是总经理。。

    摩根士丹利信息技术部门简历接收即将截止.请同学们抓紧投递 摩根士丹利9月.10月将在中国各大高校举办包含技术讲座.信息分享会以及校园宣讲会在 内的一系列校园活动.同学们将有机会和摩根士丹利高管以及返校 ...

  9. Core Animation 负责将bitmap绑定提交到 GPU-[CALayer _display]

    Core Animation 负责将bitmap绑定提交到 GPU: Core Animation一头连着CPU,一头连着GPU. ZSTest`-[ZSDTCoreTextCell drawRect ...

  10. 教你用3ds max制作多边形小狗建模

    本教程是一篇关于用3ds max来制作多边形小狗建模的简易教程,介绍地很详细,制作出来的狗很有特色,转发过来,感兴趣的朋友可以过来学习一下! 建立一个BOX,把物体放到空间原点上(这样在以后调节中间点 ...