zoukankan      html  css  js  c++  java
  • RabbitMQ指南之四:路由(Routing)和直连交换机(Direct Exchange)

    在上一章中,我们构建了一个简单的日志系统,我们可以把消息广播给很多的消费者。在本章中我们将增加一个特性:我们可以订阅这些信息中的一些信息。例如,我们希望只将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(主题模式)。

    喜欢这篇文章?欢迎打赏~~

  • 相关阅读:
    Ubuntu安装qBittorrent
    资深程序猿冒死揭开软件潜规则:无法维护的代码
    Oracle11g Active Data Guard搭建、管理
    Android 扁平化button
    Eclipse Android 代码自己主动提示功能
    Echoprint系列--编译
    一步步玩pcDuino3--mmc下的bootloader
    【Discuz】去除版权信息,标题栏与底部改动
    phoenixframe自己主动化測试平台对div弹出框(如弹出的div登陆框)的处理
    UVa
  • 原文地址:https://www.cnblogs.com/cangqinglang/p/14234363.html
Copyright © 2011-2022 走看看