4.Routing#前山翻译

注:这是RabbitMQ-java版Client的指导教程翻译系列文章,欢迎大家批评指正
第一篇Hello Word了解RabbitMQ的基本用法
第二篇Work Queues介绍队列的使用
第三篇Publish/Subscribe介绍转换器以及其中fanout类型
第四篇Routing介绍direct类型转换器
第五篇Topics介绍topic类型转换器
第六篇RPC介绍远程调用

在上一篇指导教程中,我们创建了一个日志系统,可以把日志消息广播给很多接受者。

在这篇指导教程中,我们需要添加一个功能:可以订阅消息的一部分。例如:我们会直接将严重的错误信息生成日志文件(保存在空余的磁盘上),但是依然会把所有的日志信息显示在控制台。

绑定(Bindings)

在上篇指导教程的例子中,我们已经创建过绑定的实例,你可能会觉得跟下面的代码类似:

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

绑定的含义是转换器和队列之间的一种关联,通俗来说就是一个队列对这个转换器中的消息感兴趣。

绑定可以带有一个参数:routingKey。为了避免和basic_publish中的参数产生困惑,我们将这个参数叫着binding key(绑定钥匙),下面是我们创建一个带有钥匙的绑定。

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

这个绑定钥匙的意思取决于转换器的类型,如果是我们之前使用的fanout类型转换器,那么会忽略绑定钥匙的意义。

直接转换器(Direct exchange)

在上篇指导教程中,我们的日志系统会广播消息给所有绑定转换器的消费者。现在我们扩展一下:根据消息的级别来过滤消息。举例来说,我们想一个应用只接受严重级别的消息并且写入到磁盘里,就不用浪费磁盘空间去保存警告或者信息日志的消息。

如果使用fanout转换器,那样就没有什么灵活性,不停的愚蠢的广播。

可以使用direct转换器,它的路由选择的算法是容易理解,一个消息之所以到这个队列中去,是因为队列的binding Key和发出消息的routingkey相匹配。

为了说明这个问题,看下下面的结构:

direct-exchange.png

这张结构图中,可以看到有两个队列绑定着类型为direct的转换器,第一个队列绑定钥匙为orange,第二个绑定钥匙有两个:一个是black另一个是green。

在上面的结构图中,一个带有routingkey为orange的消息发送给转换器将会被发送到队列Q1中,带有routing Key为black和green将会被发送给到队列Q2中,其他所有的消息将会被清除。

多重绑定(Multiple bindings)

direct-exchange-multiple.png

多个队列拥有相同的binding key是完全合规的,上图中我们可以在转换器x和带有bindingkey为black的队列Q1建立绑定关系。在这种情况下,direct类型的转换器具有fanout类型的一样特性,可以广播给所有匹配的队列消息。一个routingkey为black的消息将会被发送到Q1和Q2两个队列中。

发送消息(Emitting logs)

我们使用这种模型应用到日志系统上,发送给消息给direct而不是fanout类型的转换器。将以日志严重等级作为routing key。按照那种方式,消费者应用将会选择接受日志的严重等级的消息。首先我们先发送消息。

总是一样的,先声明一个转换器:

channel.exchangeDeclare(EXCHANGE_NAME, "direct");

准备好发送消息:

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

为了方面,我们假设等级分为三种:info,warning,error。

订阅(Subscribing)

只要像上篇指导教程中接受消息就可以,有一个不同的地方就是:我们可以去创建任何一个等级的绑定。

String queueName = channel.queueDeclare().getQueue();

        for(String severity : argv){

            channel.queueBind(queueName, EXCHANGE_NAME, severity);

    }

综合

python-four.png

下面是EmitLogDirect.java类,这里下载

import com.rabbitmq.client.*;

import java.io.IOException;

public class EmitLogDirect {

    private static final String EXCHANGE_NAME = "direct_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, "direct");

        String severity = getSeverity(argv);  //获取日志等级

        String message = getMessage(argv);  //获取消息

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

       System.out.println(" [x] Sent '" + severity + "':'" + message + "'");

        channel.close();

        connection.close();

        }

    //..

}

下面是ReceiveLogsDirect.java类,这里下载

import com.rabbitmq.client.*;

import java.io.IOException;

public class ReceiveLogsDirect {

    private static final String EXCHANGE_NAME = "direct_logs";

    public static void main(String[] argv) throws Exception {

        ConnectionFactory factory = new ConnectionFactory();

        factory.setHost("localhost");

        Connection connection = factory.newConnection();

        Channel channel = connection.createChannel();

        channel.exchangeDeclare(EXCHANGE_NAME, "direct");

        String queueName = channel.queueDeclare().getQueue();

        if (argv.length < 1){

        System.err.println("Usage: ReceiveLogsDirect [info] [warning] [error]");

        System.exit(1);

        }

        for(String severity : argv){

        channel.queueBind(queueName, EXCHANGE_NAME, severity);

       }

       System.out.println(" [*] Waiting for messages. To exit press CTRL+C");

    Consumer consumer = new DefaultConsumer(channel) {

        @Override

        public void handleDelivery(String consumerTag, Envelope envelope,

AMQP.BasicProperties properties, byte[] body) throws IOException {

                String message = new String(body, "UTF-8");

                System.out.println(" [x] Received '" + envelope.getRoutingKey() + "':'" + message + "'");

             }

        };

        channel.basicConsume(queueName, true, consumer);

      }

}

跟以前一样编译,运行的时候为了方面,我们使用环境便来个$CP作为路径配置:

javac -cp $CP ReceiveLogsDirect.java EmitLogDirect.java

如果你想把warning和error(而不是info)类型的日消息保存到文件中,只需要打开一个控制台和记录:

java -cp $CP ReceiveLogsDirect warning error > logs_from_rabbit.log

如果你想在你的屏幕上看到所有的日志消息,新打开一个终端和查看就可以:

java -cp $CP ReceiveLogsDirect info warning error
# => [*] Waiting for logs. To exit press CTRL+C

举个例子,发送一个error类型的日志消息:

java -cp $CP EmitLogDirect error "Run. Run. Or it will explode."
# => [x] Sent 'error':'Run. Run. Or it will explode.'

第四节的内容大致翻译完了,这里是原文链接。接着进入下一节:Topics

终篇是我对RabbitMQ使用理解的总结文章,欢迎讨教。
--谢谢--

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,542评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,596评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,021评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,682评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,792评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,985评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,107评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,845评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,299评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,612评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,747评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,441评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,072评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,828评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,069评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,545评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,658评论 2 350

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,637评论 18 139
  • 来源 RabbitMQ是用Erlang实现的一个高并发高可靠AMQP消息队列服务器。支持消息的持久化、事务、拥塞控...
    jiangmo阅读 10,353评论 2 34
  • 【译】RabbitMQ教程一 主要通过Hello Word对RabbitMQ有初步认识 【译】RabbitMQ教程...
    maxwellyue阅读 22,807评论 1 30
  • RabbitMQ笔记 本文参考资料:http://blog.csdn.net/chwshuang/article/...
    wangxiaoda阅读 2,819评论 0 11
  • 注:这是RabbitMQ-java版Client的指导教程翻译系列文章,欢迎大家批评指正第一篇Hello Word...
    前山饭店阅读 673评论 0 0