在上一章中,我们构建了一个简单的日志系统,我们可以把消息广播给很多的消费者。在本章中我们将增加一个特性:我们可以订阅这些信息中的一些信息。例如,我们希望只将error级别的错误存储到硬盘中,同时可以将所有级别(error、info、warning等)的日志都打印在控制台上。

1、绑定(Bindings)

  在上一章中,我们已经创建了绑定关系,回顾一下代码:

  1. channel.queueBind(queueName, EXCHANGE_NAME, "");

  一个绑定是一个交换器与队列之间的关系。意思是指:这个队列对这个交换器的消息感兴趣。

  该方法同时还有另一个routing Key参数,为了避免与basic_public参数产生中的路由键(routing key)混淆,我们称之为绑定键(bingind key),下面展示了如何通过一个绑定key创建一个绑定:

  1. channel.queueBind(queueName, EXCHANGE_NAME, "black");

  注意,这个绑定键(这里是"black")的含义依赖于交换器的类型。比如在我们的日志系统中,交换器类型为fanout,此时,绑定键没有任何意义,会被忽略掉。

2、直连交换机(Direct Exchange)

  在我们之前的日志系统中,所有的消息被广播给所有的消费者,但是本章的需要是希望有一个程序可以只接收error级别的日志并保存到磁盘中,而不用浪费空间去存储那些info、warning级别的日志。

  我们正在用的广播模式的交换器并不够灵活,它只是不加思索地进行广播。因此,需要使用direct exchange来代替。直连交换器的路由算法非常简单:将消息推送到binding key与该消息的routing key相同的队列。

  为了说明这点,请看下图:

  

  在该图中,直连交换器X上绑定了两个队列。第一个队列绑定了绑定键orange,第二个队列有两个绑定键:black和green。在这种场景下,一个消息在布时指定了路由键为orange将会只被路由到队列Q1,路由键为black和green的消息都将被路由到队列Q2。其他的消息都将被丢失。

3、多重绑定

  

  同一个绑定键可以绑定到不同的队列上去,在上图中,我们也可以增加一个交换器X与队列Q2的绑定键,在这种情况下,直连交换器将会和广播交换器有着相同的行为,将消息推送到所有匹配的队列。一个路由键为black的消息将会同时被推送到队列Q1和Q2。

4、发送日志

  首先我们要一如既往地创建一个交换器:

  1. channel.exchangeDeclare(EXCHANGE_NAME, "direct");

  并准备发送消息:

  1. channel.basicPublish(EXCHANGE_NAME, severity, null, message.getBytes());

  我们需要确保在我们日志系统中参数"severity"是“info”、“warning”和“error”中的一个。

5、订阅

  创建接收消息与上一章基本相同,唯一不同的是,需要在创建绑定关系时,指定severity的值:

  1. String queueName = channel.queueDeclare().getQueue();
  2.  
  3. for(String severity : argv){
  4. channel.queueBind(queueName, EXCHANGE_NAME, severity);
  5. }

6、完整的代码

  EmitLogDirect.java

  1. import com.rabbitmq.client.Channel;
  2. import com.rabbitmq.client.Connection;
  3. import com.rabbitmq.client.ConnectionFactory;
  4.  
  5. public class EmitLogDirect {
  6.  
  7. private static final String EXCHANGE_NAME = "direct_logs";
  8.  
  9. public static void main(String[] argv) throws Exception {
  10. ConnectionFactory factory = new ConnectionFactory();
  11. factory.setHost("localhost");
  12. try (Connection connection = factory.newConnection();
  13. Channel channel = connection.createChannel()) {
  14. channel.exchangeDeclare(EXCHANGE_NAME, "direct");
  15.  
  16. String severity = getSeverity(argv);
  17. String message = getMessage(argv);
  18.  
  19. channel.basicPublish(EXCHANGE_NAME, severity, null, message.getBytes("UTF-8"));
  20. System.out.println(" [x] Sent '" + severity + "':'" + message + "'");
  21. }
  22. }
  23. //..
  24. }

  ReceiveLogsDirect.java

  1. import com.rabbitmq.client.*;
  2.  
  3. public class ReceiveLogsDirect {
  4.  
  5. private static final String EXCHANGE_NAME = "direct_logs";
  6.  
  7. public static void main(String[] argv) throws Exception {
  8. ConnectionFactory factory = new ConnectionFactory();
  9. factory.setHost("localhost");
  10. Connection connection = factory.newConnection();
  11. Channel channel = connection.createChannel();
  12.  
  13. channel.exchangeDeclare(EXCHANGE_NAME, "direct");
  14. String queueName = channel.queueDeclare().getQueue();
  15.  
  16. if (argv.length < 1) {
  17. System.err.println("Usage: ReceiveLogsDirect [info] [warning] [error]");
  18. System.exit(1);
  19. }
  20.  
  21. for (String severity : argv) {
  22. channel.queueBind(queueName, EXCHANGE_NAME, severity);
  23. }
  24. System.out.println(" [*] Waiting for messages. To exit press CTRL+C");
  25.  
  26. DeliverCallback deliverCallback = (consumerTag, delivery) -> {
  27. String message = new String(delivery.getBody(), "UTF-8");
  28. System.out.println(" [x] Received '" +
  29. delivery.getEnvelope().getRoutingKey() + "':'" + message + "'");
  30. };
  31. channel.basicConsume(queueName, true, deliverCallback, consumerTag -> { });
  32. }
  33. }

  为了测试方便,我们可以把"info"、"error"、"warning"都绑定到一个队列上去,然后生产者分别往"info"、"error"、"warning"发送消息:

  此时查看RabbitMq控制台:

  

  到此,发布-订阅涉及到的相关知识点都讲解完了,下一章将讲解Topic(主题模式)。

RabbitMQ指南之四:路由(Routing)和直连交换机(Direct Exchange)的更多相关文章

  1. RabbitMQ入门:路由(Routing)

    在上一篇博客<RabbitMQ入门:发布/订阅(Publish/Subscribe)>中,我们认识了fanout类型的exchange,它是一种通过广播方式发送消息的路由器,所有和exch ...

  2. RabbitMQ入门(4)——路由(Routing)

    这一篇我们将介绍如何订阅消息的一个子集.例如,我们只需要将日志中的error消息存储到日志文件中而将所有日志消息都在控制台打印出来. 绑定(Bindings) 在前面的例子中,我们创建了交换机和队列的 ...

  3. Rabbit的直连交换机direct

    直连交换机类型为:direct.加入了路由键routingKey的概念. 就是说 生产者投递消息给指定交换机的指定路由键. 只有绑定了此交换机指定路由键的消息队列才可以收到消息. 生产者: packa ...

  4. PHP 下基于 php-amqp 扩展的 RabbitMQ 简单用例 (一) -- 安装 AMQP 扩展和 Direct Exchange 模式

    Windows 安装 amqp 扩展 RabbitMQ 是基于 amqp(高级消息队列协议) 协议的.使用 RabbitMQ 前必须为 PHP 安装相应的 amqp 扩展. 下载相应版本的 amqp ...

  5. RabbitMQ --- 直连交换机 【 有回调方法,获取消费结果 】

    1.前言 上一随笔详细记录了直连交换机的方法,发送的消息是异步的,如果消息未被消费者消费,那么可以一直存在消息队列中. 那么有没有办法做一个回调,当消息被消费后,被通知消息成功被消费者消费啦? 答案是 ...

  6. RabbitMQ --- 直连交换机 【 无回调方法,不能获取消费结果 】

    1.前言 消息队列除了kafka 外,还有许多种,比如RabbitMQ .ActiveMQ.ZeroMQ.JMQ等. 老牌的ActiveMQ ,底层使用Java写的,资源消耗大,速度也慢,但是适合 J ...

  7. spring boot 集成 rabbitmq 指南

    先决条件 rabbitmq server 安装参考 一个添加了 web 依赖的 spring boot 项目 我的版本是 2.5.2 添加 maven 依赖 <dependency> &l ...

  8. RabbitMQ入门教程——路由(Routing)

    绑定( Bindings)   之前的文章中我们已经创建过bindings,代码如下:         channel.QueueBind(queue: queueName, exchange: EX ...

  9. RabbitMQ官方中文入门教程(PHP版) 第四部分:路由(Routing)

    路由(Routing) 在前面的教程中,我们实现了一个简单的日志系统.可以把日志消息广播给多个接收者. 本篇教程中我们打算新增一个功能——使得它能够只订阅消息的一个字集.例如,我们只需要把严重的错误日 ...

随机推荐

  1. Mycat集群方案收集(待实践)

    先收集,后续再实践. 我想,市面上开源方案中,涉及到高可用和负载均衡的部署,无论是哪一个产品应用,都基本离不开LVS+Keepalived+HAProxy+Nginx等等. 下面是收集的教程: htt ...

  2. 程序猿Web面试之JSON

     JSON是什么? JSON(JavaScript对象表示法), 是在网络通信下.经常使用的一种数据表达格式,它有助于我们于一个自描写叙述的,独立的和轻的方式呈现并交换数据. 这些数据能够易于和转 ...

  3. 【python自制】让大白成为你的个人助手!

    我做这个软件就是要让卖萌进行究竟! 官方站点:http://www.jackeriss.com/companions.htm GitHub:https://github.com/Jackeriss/C ...

  4. python 区块链程序

    python 区块链程序 学习了:https://mp.weixin.qq.com/s?__biz=MzAxODcyNjEzNQ==&mid=2247484921&idx=1& ...

  5. C++ RTTI介绍

    一.定义:RTTI:Run Time Type Identification ,执行时类型识别:指程序可以使用基类的指针或引用来检索其所指对象的实际派生类型. 二.使用方式:C++中有两个操作符提供R ...

  6. 怎样在OTN站点高速找到asm包并下载 (Oracle RAC)

    怎样在OTN站点高速找到asm包并下载 ***********************************************声明******************************* ...

  7. iOS动画案例(1) 仿qq账号信息里的一个动画

       受人所托,模仿qq账号信息里的一个动画,感觉挺有意思,也没感觉有多难,就开始做了,结果才发现学的数学知识都还给体育老师了,研究了大半天才做出来.    先看一下动画效果:   用到的知识点: ( ...

  8. AVD那些事儿

    启动了AVD却说找不到AVD 错误提示: No active compatible AVD's or devices found. Relaunch this configuration after ...

  9. c++运算符重载以及一些基本概念

    c++primer第四版435 1.赋值( = ), 下标( [ ] ) ,调用 (  ( )  ), 成员訪问箭头 (->)等操作符必须定义为成员,定义为非成员时,编译器报错 2. 像赋值一样 ...

  10. 【uoj35】后缀排序

    后缀数组模板题 #include<algorithm> #include<cstdlib> #include<cstring> #include<cstdio ...