为了实现一个新功能:只订阅消息的一个子集,例如只需要把严重的错误日志信息写入日志文件(存储到磁盘上),但同时仍然把所有的日志信息输出到控制台中。

绑定(Bindings)

创建绑定

channel.queue_bind(exchange=exchange_name,
queue=queue_name)

建立exchange和queue之间的关系,简单理解是这个队列对这个交换器的消息感兴趣。

绑定的时候可以带上一个额外的routing_key参数,为了避免与basic_publish参数混淆,叫做binding key.

创建一个带binding key的绑定

channel.queue_bind(exchange=exchange_name,
queue=queue_name,
routing_key='black')

binding key的含义取决于交换器的类型,对于fanout类型会忽略这个值。

Driect类型的交换器exchange

使用fanout类型的交换器exchange是广播类型,对于消息的过滤需要使用direct类型

使用direct类型的交换器,交换器将会对binding key和routing key进行精确匹配,从而确定消息该分发到哪个队列。

如上图,可以看到x交换器和两个队列进行绑定,第一个队列使用orange作为binding key,第二个队列有两个绑定,一个使用black,一个使用green。

这样,当routing key为orange的消息发布到交换器,会路由到队列Q1,black和green两个类型路由Q2,其他的所有消息将会被丢弃。

多个绑定(Multiple bindings)

多个队列使用相同的binding key是合法的。

可以添加一个x和Q1之间的绑定,也可以再添加一个x和Q2的绑定。这样以来,指定交换direct类型和fanout广播类型功能相同。

Emmiting logs

将会发送消息到一个direct exchange,把日志级别作为routing key。

这样负责处理接收的脚本可以选择要处理的日志级别。

创建一个direct类型的交换器

channel.exchange_declare(exchange='direct_logs',
type='direct')

然后发送一条消息:

channel.basic_publish(exchange='direct_logs',
routing_key=severity,
body=message)

severity值假定为info,warning,error中的一个

订阅(subscribing)

result = channel.queue_declare(exclusive=True)
queue_name = result.method.queue for severity in severities:
channel.queue_bind(exchange='direct_logs',
queue=queue_name,
routing_key=severity)

为每一个日志级别创建一个新的绑定。

例子

将error信息发送到一个队列,将所有信息发送另一个队列

emit_log

#!/usr/bin/env python
#-*- coding:utf8 -*-
import sys
import pika
import logging logging.basicConfig(format='%(levelname)s:$(message)s',level=logging.CRITICAL) def emit_log(): pika.connection.Parameters.DEFAULT_HOST = 'localhost'
pika.connection.Parameters.DEFAULT_PORT = 5672
pika.connection.Parameters.DEFAULT_VIRTUAL_HOST = '/'
pika.connection.Parameters.DEFAULT_USERNAME = 'guosong'
pika.connection.Parameters.DEFAULT_PASSWORD = 'guosong' para = pika.connection.Parameters() connection = pika.BlockingConnection(para) channel = connection.channel()
#声明一个direct_logs交换器,类型为direct
channel.exchange_declare(exchange='direct_logs',type='direct') #指定日志级别
serverity = sys.argv[1] if len(sys.argv) >1 else 'info' message = '.'.join(sys.argv[1:]) or "info:Hello World!" #发送的时候指定routing_key为空,没有绑定队列到交换器上,消息将会丢失
#对于日志类消息,如果没有消费者监听的话,这些消息就会忽略
channel.basic_publish(exchange='logs',routing_key=serverity,body=message) #%r也是string类型
print "[x] Sent %r" % (message,) connection.close() if __name__ == '__main__':
emit_log()

 接收者

#!/usr/bin/env python
#-*- coding:utf8 -*-
import sys
import pika
import logging logging.basicConfig(format='%(levelname)s:$(message)s',level=logging.CRITICAL) #回调函数,处理消息
def callback(ch, method, properties, body):
print " [x] %r " % (body,) def receive_logs():
pika.connection.Parameters.DEFAULT_HOST = 'localhost'
pika.connection.Parameters.DEFAULT_PORT = 5672
pika.connection.Parameters.DEFAULT_VIRTUAL_HOST = '/'
pika.connection.Parameters.DEFAULT_USERNAME = 'guosong'
pika.connection.Parameters.DEFAULT_PASSWORD = 'guosong' para = pika.connection.Parameters()
connection = pika.BlockingConnection(para)
channel = connection.channel() #声明一个logs交换器,类型为fanout,不允许发布消息到不存在的交换器
channel.exchange_declare(exchange='direct_logs',type='direct') #声明一个随机队列,设置exclusive=True,在该consumer退出的时候,对应的队列被删除
result = channel.queue_declare(exclusive=True)
#获取随机队列的名称
queue_name = result.method.queue serverities = sys.argv[1:]
if not serverities:
print >> sys.stderr, "Usage: %s [info] [warning] [error]" % sys.argv[0]
sys.exit(1) for serverity in serverities:
#绑定交换器和队列
channel.queue_bind(exchange='logs',queue=queue_name,routing_key=serverity)
print '[*] Wating for logs.To exit press CTRL+C' #开始消费消息
channel.basic_consume(callback,queue=queue_name,no_ack=True)
channel.start_consuming() if __name__ == '__main__':
receive_logs()

如果只希望保存warning和error级别的日志到磁盘,只需要打开控制台并输入:

guosong@guosong:~/code/rabbitmq/ch4$ ./receive_logs.py warning error >logs

如果需要所有的话,执行如下命令:

guosong@guosong:~/code/rabbitmq/ch4$ ./receive_logs.py info warning error >logs

  

RabbitMQ之路由的更多相关文章

  1. RabbitMQ之路由(Routing)【译】

    在上一节中,我们创建了一个简单的日志系统,可以广播消息到很多接收者. 这一节,我们将在上一节的基础上加一个功能--订阅部分消息.例如,我们只将严重错误信息写入到日志文件保存在磁盘上,同时我们能将所有的 ...

  2. 【译】RabbitMQ:路由(Routing)

    在前一篇中,我们构建了一个简单的日志系统,我们已经能够广播消息到许多的接收者.在这一篇中,我们希望增加一个特性,让订阅消息的子集成为可能.例如,我们可以将重要的错误日志存放到日志文件(即,磁盘上面), ...

  3. RabbitMQ入门-路由-有选择的接受消息

    比如一个日志系统,之前的处理方式呢,是各种类型(info,error,warning)的消息都发给订阅者,可是实际情况上不一定都需要.可能A需要error,其他的都不需要.那么就引入了今天的处理方式- ...

  4. RabbitMQ之路由键转发消息

    RabbitMQ学习 参考:https://www.jianshu.com/p/6b62a0ed2491 消息队列:目前流行的有 Kafka.RabbitMQ.ActiveMQ等 功能:为了解决消息的 ...

  5. Rabbitmq(5) 路由模式

    设置路由键 发送者 package com.aynu.bootamqp.service; import com.aynu.bootamqp.commons.utils.Amqp; import com ...

  6. RabbitMQ 的路由模式 Topic模式

    模型 生产者 package cn.wh; import java.io.IOException; import java.util.concurrent.TimeoutException; impo ...

  7. 五.RabbitMQ之路由(Routing)和主题(topics)

    翻译官网的文章已经翻译了几天了,这份官方文档写的总体算是很简洁易懂.它让我们很快的入门并了解了RabbitMQ的运作原理和使用方式.本篇最后介绍一下Exchange的另外两种类别,即direct和to ...

  8. rabbitmq (四) 路由

    上文讲的是广播类型fanout 本章讲 direct和topic. 当使用广播类型fanout的时候: routingKey字段不起作用. direct:精确匹配 routingKey:匹配一个单词, ...

  9. .Net使用RabbitMQ详解

    序言 这里原来有一句话,触犯啦天条,被阉割!!!! 首先不去讨论我的日志组件怎么样.因为有些日志需要走网络,有的又不需要走网路,也是有性能与业务场景的多般变化在其中,就把他抛开,我们只谈消息Rabbi ...

随机推荐

  1. 【转】 Python调用(运行)外部程序

    在Python中可以方便地使用os模块运行其他的脚本或者程序,这样就可以在脚本中直接使用其他脚本,或者程序提供的功能,而不必再次编写实现该功能的代码.为了更好地控制运行的进程,可以使用win32pro ...

  2. SAP 邮件发送

    1.******************** *调用发送邮件函数 CALL FUNCTION 'SO_NEW_DOCUMENT_ATT_SEND_API1' EXPORTING DOCUMENT_DA ...

  3. win10 UWP 申请微软开发者

    申请微软开发者可以到https://dev.windows.com/zh-cn/programs/join 如果是学生,先去http://www.dreamspark.com/ 如果是英文,点stud ...

  4. PyCharm 2017 免费 破解 注册 激活 教程(附 License Server 地址)(Python 编辑器 IDE 推荐)

    许多朋友都在问如何破解 PyCharm 2017 Professional 专业版,咪博士对此是坚决反对的! 不到万不得已,请不要这样做.破解之前,请拖到文章末尾,思考几个问题,想明白你确实需要这样做 ...

  5. Lustre文件系统测试——obdfilter-survey测试

    Lustre文件系统测试--obdfilter-survey测试 介绍 该测试主要是在lustre文件系统工作环境下进行,将直接在ost上生成工作负载模拟并行文件访问,可准确检测盘阵在lustre文件 ...

  6. 【转】NOR Flash擦写和原理分析

    1. NOR FLASH 的简单介绍 NOR FLASH 是很常见的一种存储芯片,数据掉电不会丢失.NOR FLASH支持Execute On Chip,即程序可以直接在FLASH片内执行(这意味着存 ...

  7. [ACdream] 女神教你字符串——三个气球

    Problem Description 女神邀请众ACdream开联欢会,显然作为ACM的佼佼者,气球是不能少的~.女神准备了三种颜色的气球,红色,黄色,绿色(交通信号灯?) 有气球还不能满足女神,女 ...

  8. LeetCode 170. Two Sum III - Data structure design (两数之和之三 - 数据结构设计)$

    Design and implement a TwoSum class. It should support the following operations: add and find. add - ...

  9. 异常详细信息: Abp.AbpException: No language defined!

    程序运行后,出现错误:No language defined! 解决方法: 1.检查是否已创建数据库,若未创建则在程序包管理控制台执行命令:Update-Database 2.检查表AbpLangua ...

  10. 暑假练习赛 006 B Bear and Prime 100

    Bear and Prime 100Crawling in process... Crawling failed Time Limit:1000MS     Memory Limit:262144KB ...