RabbitMQ学习总结(5)——发布和订阅实例详解
2019独角兽企业重金招聘Python工程师标准>>>
一、Publish/Subscribe(发布/订阅)(using the Java Client)
在前面的教程中,我们创建了一个work Queue(工作队列)。工作队列背后的假设是每个任务是交付给一个工作者(worker) 也就是均匀分给每个消费者。在本部分,我们将做一些完全不同的事情,我们将提供一个消息到多个消费者。这种模式被称为“发布/订阅”。
为了说明这个模式,我们将构建一个简单的日志系统。它将包括两个项目:
第一个将发出日志消息 第二个将接收并打印它们。
在我们的日志系统,每运行一次,接收器项目将得到消息的副本。这样我们能够运行一个接收机并且可以直接记录到磁盘,同时我们可以运行另一个接收器,看到屏幕上的日志。 注:从本质上讲,发表日志消息广播给所有的接收者。
下面让我们脑中带几个问题,让我们一步一步去解决:
如果我把消息分配给所有的消费者,我们将怎么做呢?
二、Exchanges(交换机)
- 生产者是一个用户发送消息的应用程序。
- 一个队列是存储消息的缓冲区。
- 消费者是一个用户应用程序接收消息。
RabbitMQ的消息模型的核心思想是,生产者从未直接向队列发送任何消息。实际上,经常生产者甚至不知道消息是否会被运送到任何队列。
相反,生产者只能发送Exchanges (消息交换区)。交换是一个非常简单的事情。 一方面它从生产者那收到消息并推他们到另一边队列。交换区必须知道如何处理它收到一条消息:
- 它应该被加到一个特定的队列吗?
- 它应该被加到多队列?
- 或者它应该丢弃吗?
channel.exchangeDeclare("logs", "fanout");
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.
channel.basicPublish("", "hello", null, message.getBytes());
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(发布者/订阅者 实现)
- </pre><pre name="code" class="java">import java.io.IOException;
- import com.rabbitmq.client.ConnectionFactory;
- import com.rabbitmq.client.Connection;
- import com.rabbitmq.client.Channel;
- public class EmitLog {
- private static final String EXCHANGE_NAME = "logs";
- public static void main(String[] argv)
- throws java.io.IOException {
- ConnectionFactory factory = new ConnectionFactory();
- factory.setHost("localhost");
- Connection connection = factory.newConnection();
- Channel channel = connection.createChannel();
- channel.exchangeDeclare(EXCHANGE_NAME, "fanout");
- String message = getMessage(argv);
- channel.basicPublish(EXCHANGE_NAME, "", null, message.getBytes());
- System.out.println(" [x] Sent '" + message + "'");
- channel.close();
- connection.close();
- }
- //...
- }
接收端:
- import java.io.IOException;
- import com.rabbitmq.client.ConnectionFactory;
- import com.rabbitmq.client.Connection;
- import com.rabbitmq.client.Channel;
- import com.rabbitmq.client.QueueingConsumer;
- public class ReceiveLogs {
- private static final String EXCHANGE_NAME = "logs";
- public static void main(String[] argv)
- throws java.io.IOException,
- java.lang.InterruptedException {
- ConnectionFactory factory = new ConnectionFactory();
- factory.setHost("localhost");
- Connection connection = factory.newConnection();
- Channel channel = connection.createChannel();
- channel.exchangeDeclare(EXCHANGE_NAME, "fanout");
- String queueName = channel.queueDeclare().getQueue();
- channel.queueBind(queueName, EXCHANGE_NAME, "");
- System.out.println(" [*] Waiting for messages. To exit press CTRL+C");
- QueueingConsumer consumer = new QueueingConsumer(channel);
- channel.basicConsume(queueName, true, consumer);
- while (true) {
- QueueingConsumer.Delivery delivery = consumer.nextDelivery();
- String message = new String(delivery.getBody());
- System.out.println(" [x] Received '" + message + "'");
- }
- }
- }
像以前一样,我们开始做编译
$ 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.
转载于:https://my.oschina.net/zhanghaiyang/blog/592607
RabbitMQ学习总结(5)——发布和订阅实例详解相关推荐
- 入门学习Linux常用必会命令实例详解
Linux提供了大量的命令,利用它可以有效地完成大量的工作,如磁盘操作.文件存取.目录操作.进程管理.文件权限设定等.所以,在Linux系统上工作离不开使用系统提供的命令.要想真正理解Linux系统, ...
- RabbitMQ学习笔记:消息追踪Firehose功能详解
在使用任何消息中间件的过程中,难免会出现消息异常丢失的情况.对于RabbitMQ而言,可能是生产者与Broker断开了连接并且没有任何重试机制:也可能是消费者在处理消息时发生了异常,不过却提前进行了a ...
- StackExchange.Redis学习笔记(五) 发布和订阅
StackExchange.Redis学习笔记(五) 发布和订阅 原文:StackExchange.Redis学习笔记(五) 发布和订阅 Redis命令中的Pub/Sub Redis在 2.0之后的版 ...
- java 控制jsp_JSP学习之Java Web中的安全控制实例详解
普通用户界面 修改登录的Servlet,修改后的代码如下: LoginProcess.java代码: package servlet; import javabean.User; import jav ...
- expect学习笔记及实例详解【转】
1. expect是基于tcl演变而来的,所以很多语法和tcl类似,基本的语法如下所示: 1.1 首行加上/usr/bin/expect 1.2 spawn: 后面加上需要执行的shell命令,比如说 ...
- java学习 类变量 类方法_这篇文章主要介绍了JAVA类变量及类方法代码实例详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下类变量(...
这篇文章主要介绍了JAVA类变量及类方法代码实例详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 类变量(static) 类变量是该类的所有对象共 ...
- mysql数据库select语句用法_mysql学习笔记之完整的select语句用法实例详解
本文实例讲述了mysql学习笔记之完整的select语句用法.分享给大家供大家参考,具体如下: 本文内容: 完整语法 去重选项 字段别名 数据源 where group by having order ...
- 官网实例详解-目录和实例简介-keras学习笔记四
https://github.com/keras-team/keras/tree/master/examples Keras examples directory Keras实例目录 (点击跳转) 官 ...
- 《ABAQUS有限元分析实例详解》学习笔记_51CAE_新浪博客
石亦平老师的<ABAQUS有限元分析实例详解>当属ABAQUS学习的经典著作,一边认真拜读一边在此写下点滴笔记,供自己参考. 1.ABAQUS/CAE并没有自己专用的量纲系统,用户建立的整 ...
最新文章
- lua学习笔记-HelloWorld
- 零位扩展和符号位扩展
- behavior php,behavior.php
- 自定义注解,aop实现注解锁
- Spark运行原理剖析
- 带日期的bean转为json(bean-JSON)
- jvm 加载class文件过程
- 2001年分区联赛提高组之二_数的划分_ssl1016_dfs
- 为什么有的人喜欢夜间工作_为什么开发人员喜欢在夜间编码
- Python OOP:继承、单继承、多继承、__mro__、子类重写父类同名属性和方法、子类调用父类同名属性和方法、多层继承、super()、私有(实例)属性和方法、获取修改私有属性值、私有类属性
- mysql binlog 备份_Mysql数据库的增量备份与还原
- c语言static知识点,C语言知识点集锦
- Nginx安装与常用配置
- Python selenium 滚动页面以及滚动至元素可见(转载)
- android自定义拨号键盘,Android拨号键盘增加魔力爱心数字
- Ubuntu16 网卡rtl8723be 驱动安装
- 字符串“false“转 Boolean false
- 03 CarbonData的数据管理(建表)
- 什么是微型计算机组成原理,SJTU 《微型计算机组成原理》备考题
- EJS References
热门文章
- 【第三组】用例+功能说明+技术说明
- C语言 · 计算时间
- 2014家电盘点:求变与创新
- Android布局整合include界面控件 示例
- PHP 简单的数字过滤函数
- eplan模板_EPLAN之3D箱柜清单自动生成
- android activity根节点addview_Activity问你4个问题,你敢回答吗?
- axurehtml打开不用用_还有人花钱买会员看剧?赶紧用iPhone免费追剧
- hadoop 爬虫_python爬虫知识点梳理:带你全面入门python爬虫
- Wireshark数据抓包教程之Wireshark捕获数据