转自 http://blog.csdn.net/xiaoxian8023/article/details/48733249

翻译地址:http://www.rabbitmq.com/tutorials/tutorial-four-java.html

前篇博文中,我们建立了一个简单的日志系统。可以广播消息给多个消费者。本篇博文,我们将添加新的特性——我们可以只订阅部分消息。比如:我们可以接收Error级别的消息写入文件。同时仍然可以在控制台打印所有日志。

Bindings(绑定)

在上一篇博客中我们已经使用过绑定。类似下面的代码:

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

绑定表示转换器与队列之间的关系。可以简单的人为:队列对该转发器上的消息感兴趣。

绑定可以设定额外的routingKey参数。为了与避免basicPublish方法(发布消息的方法)的参数混淆,我们准备把它称作绑定键(binding key)。下面展示如何使用绑定键(binding key)来创建一个绑定:

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

绑定键关键取决于转换器的类型。对于fanout类型,忽略此参数。

Direct exchange(直接转发)

前面讲到我们的日志系统广播消息给所有的消费者。我们想对其扩展,根据消息的严重性来过滤消息。例如:我们希望将致命错误的日志消息记录到文件,而不是把磁盘空间浪费在warn和info类型的日志上。我们使用的fanout转发器,不能给我们太多的灵活性。它仅仅只是盲目的广播而已。我们使用direct转发器进行代替,其背后的算法很简单——消息会被推送至绑定键(binding
key)和消息发布附带的选择键(routing key)完全匹配的队列。

在上图中,我们可以看到direct类型的转发器与2个队列进行了绑定。第一个队列使用的绑定键是orange,第二个队列绑定键为black和green。这样当消息发布到转发器是,附带orange绑定键的消息将被路由到队列Q1中去。附带black和green绑定键的消息被路由到Q2中去。其他消息全部丢弃。

Multiple bindings(多重绑定)

使用一个绑定键绑定多个队列是完全合法的。如上图,绑定键black绑定了2个队列——Q1和Q2。

Emitting logs(发送日志)

我们将这种模式用于日志系统,发送消息给direct类型的转发器。我们将 提供日志严重性做为绑定键。那样,接收程序可以选择性的接收严重性的消息。首先关注发送日志的代码:

像往常一样首先创建一个转换器:

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

然后为发送消息做准备:

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

为了简化代码,我们假定日志的严重性是‘info’,‘warning’,‘error’中之一。

Subscribing(订阅)

接收消息跟前面博文中的一样。我们仅需要修改一个地方:为每一个我们感兴趣的严重性的消息,创建一个新的绑定。

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

完整的例子

发送端代码(EmitLogDirect.java)

  1. public class EmitLogDirect {
  2. private final static String EXCHANGE_NAME = "direct_logs";
  3. public static void main(String[] args) throws IOException {
  4. /**
  5. * 创建连接连接到MabbitMQ
  6. */
  7. ConnectionFactory factory = new ConnectionFactory();
  8. // 设置MabbitMQ所在主机ip或者主机名
  9. factory.setHost("127.0.0.1");
  10. // 创建一个连接
  11. Connection connection = factory.newConnection();
  12. // 创建一个频道
  13. Channel channel = connection.createChannel();
  14. // 指定转发——广播
  15. channel.exchangeDeclare(EXCHANGE_NAME, "direct");
  16. //所有日志严重性级别
  17. String[] severities={"error","info","warning"};
  18. for(int i=0;i<3;i++){
  19. String severity = severities[i%3];//每一次发送一条不同严重性的日志
  20. // 发送的消息
  21. String message = "Hello World"+Strings.repeat(".", i+1);
  22. //参数1:exchange name
  23. //参数2:routing key
  24. channel.basicPublish(EXCHANGE_NAME, severity, null, message.getBytes());
  25. System.out.println(" [x] Sent '" + severity +"':'"+ message + "'");
  26. }
  27. // 关闭频道和连接
  28. channel.close();
  29. connection.close();
  30. }
  31. }

消费者1(ReceiveLogs2Console.java)

  1. public class ReceiveLogs2Console {
  2. private static final String EXCHANGE_NAME = "direct_logs";
  3. public static void main(String[] argv) throws IOException, InterruptedException {
  4. ConnectionFactory factory = new ConnectionFactory();
  5. factory.setHost("127.0.0.1");
  6. // 打开连接和创建频道,与发送端一样
  7. Connection connection = factory.newConnection();
  8. final Channel channel = connection.createChannel();
  9. channel.exchangeDeclare(EXCHANGE_NAME, "direct");
  10. // 声明一个随机队列
  11. String queueName = channel.queueDeclare().getQueue();
  12. //所有日志严重性级别
  13. String[] severities={"error","info","warning"};
  14. for (String severity : severities) {
  15. //关注所有级别的日志(多重绑定)
  16. channel.queueBind(queueName, EXCHANGE_NAME, severity);
  17. }
  18. System.out.println(" [*] Waiting for messages. To exit press CTRL+C");
  19. // 创建队列消费者
  20. final Consumer consumer = new DefaultConsumer(channel) {
  21. @Override
  22. public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
  23. String message = new String(body, "UTF-8");
  24. System.out.println(" [x] Received '"  + envelope.getRoutingKey() + "':'" + message + "'");
  25. }
  26. };
  27. channel.basicConsume(queueName, true, consumer);
  28. }
  29. }

消费者2(ReceiveLogs2File.java)

  1. public class ReceiveLogs2File {
  2. private static final String EXCHANGE_NAME = "direct_logs";
  3. public static void main(String[] argv) throws IOException, InterruptedException {
  4. ConnectionFactory factory = new ConnectionFactory();
  5. factory.setHost("127.0.0.1");
  6. // 打开连接和创建频道,与发送端一样
  7. Connection connection = factory.newConnection();
  8. final Channel channel = connection.createChannel();
  9. channel.exchangeDeclare(EXCHANGE_NAME, "direct");
  10. // 声明一个随机队列
  11. String queueName = channel.queueDeclare().getQueue();
  12. String severity="error";//只关注error级别的日志,然后记录到文件中去。
  13. channel.queueBind(queueName, EXCHANGE_NAME, severity);
  14. System.out.println(" [*] Waiting for messages. To exit press CTRL+C");
  15. // 创建队列消费者
  16. final Consumer consumer = new DefaultConsumer(channel) {
  17. @Override
  18. public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {
  19. String message = new String(body, "UTF-8");
  20. //记录日志到文件:
  21. print2File( "["+ envelope.getRoutingKey() + "] "+message);
  22. }
  23. };
  24. channel.basicConsume(queueName, true, consumer);
  25. }
  26. private static void print2File(String msg) {
  27. try {
  28. String dir = ReceiveLogs2File.class.getClassLoader().getResource("").getPath();
  29. String logFileName = new SimpleDateFormat("yyyy-MM-dd").format(new Date());
  30. File file = new File(dir, logFileName + ".log");
  31. FileOutputStream fos = new FileOutputStream(file, true);
  32. fos.write((new SimpleDateFormat("HH:mm:ss").format(new Date())+" - "+msg + "\r\n").getBytes());
  33. fos.flush();
  34. fos.close();
  35. } catch (FileNotFoundException e) {
  36. e.printStackTrace();
  37. } catch (IOException e) {
  38. e.printStackTrace();
  39. }
  40. }
  41. }

最终结果:

       罗哩罗嗦的说这么多,其实就是说了这么一件事:我们可以使用Direct exchange+routingKey来过滤自己感兴趣的消息。一个队列可以绑定多个routingKey。这就是我们今天的主题——路由选择。

轻松搞定RabbitMQ(五)——路由选择的更多相关文章

  1. 轻松搞定RabbitMQ(六)——主题

    转自 http://blog.csdn.net/xiaoxian8023/article/details/48806871 翻译地址:http://www.rabbitmq.com/tutorials ...

  2. 轻松搞定RabbitMQ(二)——工作队列之消息分发机制

    转自 http://blog.csdn.net/xiaoxian8023/article/details/48681987 上一篇博文中简单介绍了一下RabbitMQ的基础知识,并写了一个经典语言入门 ...

  3. 轻松搞定RabbitMQ(一)——RabbitMQ基础知识+HelloWorld

    转自 http://blog.csdn.net/xiaoxian8023/article/details/48679609 本文是简单介绍一下RabbitMQ,参考官网上的教程.同时加入了一些自己的理 ...

  4. 轻松搞定RabbitMQ(四)——发布/订阅

    转自 http://blog.csdn.net/xiaoxian8023/article/details/48729479 翻译地址:http://www.rabbitmq.com/tutorials ...

  5. 轻松搞定RabbitMQ(三)——消息应答与消息持久化

    转自 http://blog.csdn.net/xiaoxian8023/article/details/48710653 这个官网的第二个例子中的消息应答和消息持久化部分.我把它摘出来作为单独的一块 ...

  6. 【转】轻松搞定FTP之FlashFxp全攻略

    转载网址:http://www.newhua.com/2008/0603/39163.shtml 轻松搞定FTP之FlashFxp全攻略 导读: FlashFXP是一款功能强大的FXP/FTP软件,融 ...

  7. Webcast / 技术小视频制作方法——自己动手录制video轻松搞定

    Webcast / 技术小视频制作方法——自己动手录制video轻松搞定 http://blog.sina.com.cn/s/blog_67d387490100wdnh.html 最近申请加入MSP的 ...

  8. 【微服务】之五:轻松搞定SpringCloud微服务-调用远程组件Feign

    上一篇文章讲到了负载均衡在Spring Cloud体系中的体现,其实Spring Cloud是提供了多种客户端调用的组件,各个微服务都是以HTTP接口的形式暴露自身服务的,因此在调用远程服务时就必须使 ...

  9. 使用BleLib的轻松搞定Android低功耗蓝牙Ble 4.0开发具体解释

    转载请注明来源: http://blog.csdn.net/kjunchen/article/details/50909410 使用BleLib的轻松搞定Android低功耗蓝牙Ble 4.0开发具体 ...

随机推荐

  1. POJ 1315 Don't Get Rooked

    Don't Get Rooked Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 2086   Accepted: 1325 ...

  2. ThinkPHP3.2 杂记

    如果父类有_initialize()方法,子类用任何方法前都会调用父类的 _initialize方法,避免这种现象可以在子类中声明_initialize()方法,可以控制是否调用父类的_initial ...

  3. 许式伟看 Facebook 发币(上): 区块链, 比特币与 Libra 币

    你好,我是七牛云许式伟. Facebook(脸书)于6月18日发布了其加密数字货币项目白皮书.该数字货币被命名为 Libra(天秤座),象征着平衡与公正.此前,BBC 报道说这个数字货币叫 Globa ...

  4. 基于 K8S 构建数据中心操作系统

    在 12 月 22 日 ECUG 的下午场 ,七牛云容器计算部技术总监袁晓沛为大家带来了主题为<基于 K8S 的 DCOS 之路>的精彩分享,向大家介绍了七牛容器云目前 K8S 的状况和产 ...

  5. 【bzoj4816】[Sdoi2017]数字表格 莫比乌斯反演

    题目描述 Doris刚刚学习了fibonacci数列.用f[i]表示数列的第i项,那么 f[0]=0 f[1]=1 f[n]=f[n-1]+f[n-2],n>=2 Doris用老师的超级计算机生 ...

  6. 怎样抓获或忽略像control-C这样的键盘中断?

    基本步骤是调用signal():#include <signal.h>singal(SIGINT, SIG_IGN); 就可以忽略中断信号, 或者:extern void func(int ...

  7. Manjaro中源码安装gcc7.1

    刚刚gcc 7.1也出来了,想在使用熟悉的linux下试试,特记录如下: 准备必要的系统环境:(升级系统到最新,安装必要的工具) pacman -Syyu                        ...

  8. Adoquery的 moveby和GotoBookmark,RecNo

    GotoBookmark 是必须存在的记录,再次返回原来那个记录的位置,但是原来的那个记录必须存在,所以不适合[删除订单后回到原来的位置],因为原来的订单已经不存在了,删除了, moveby(),从当 ...

  9. 开发使用mysql的一些必备知识点整理(三)高级

    简介 实体与实体之间有3种对应关系,这些关系也需要存储下来 在开发中需要对存储的数据进行一些处理,用到内置的一些函数 视图用于完成查询语句的封装 事务可以保证复杂的增删改操作有效 关系 创建成绩表sc ...

  10. AC日记——一元三次方程求解 洛谷 P1024

    题目描述 有形如:ax3+bx2+cx+d=0 这样的一个一元三次方程.给出该方程中各项的系数(a,b,c,d 均为实数),并约定该方程存在三个不同实根(根的范围在-100至100之间),且根与根之差 ...