RabbitMQ 专栏七 RabbitMQ - Topic

RabbitMQ入门教程 For Java【5】 - Topic

2016年01月14日 16:26:34


RabbitMQ入门教程 For Java【5】 - Topic



我的开发环境:

操作系统: Windows7 64bit 

开发环境: JDK 1.7 - 1.7.0_55

开发工具: Eclipse Kepler SR2

RabbitMQ版本:  3.6.0

Elang版本: erl7.2.1

关于Windows7下安装RabbitMQ的教程请先在网上找一下,有空我再补安装教程。

源码地址

https://github.com/chwshuang/rabbitmq.git



Topic模式

        匹配模式,如果按照百度翻译和百度百科,直接叫主题或者话题就得了,但是如果你真的明白它在RabbitMQ中代表什么,就不能这么直接的翻译成中文了。如果要用中文理解它的意思,先了解它在RabbitMQ中用来做什么:topic类型的交换器允许在RabbitMQ中使用模糊匹配来绑定自己感兴趣的信息。

        所以,我觉得这一章应该叫macth模式更合适,中文 - 匹配模式。

        在上一章,通过直连交换器,生产者发送不同路由关键字的日志,消费者端通过绑定自己感兴趣的路由关键字来接收消息,进行完善日志系统。如果我想只接收生产者com.test.rabbitmq.topic包下的日志,其他包的忽略掉,之前的日志系统处理起来可能就非常麻烦,还好,我们有匹配模式,现在我们将生产者发送过来的消息按照包名来命名,那么消费者端就可以在匹配模式下使用【#.topic.*】这个路由关键字来获得感兴趣的消息。

 

匹配交换器

通过匹配交换器,我们可以配置更灵活的消息系统,你可以在匹配交换器模式下发送这样的路由关键字:

“a.b.c”、“c.d”、“quick.orange.rabbit”

不过一定要记住,路由关键字【routingKey】不能超过255个字节(bytes)

匹配交换器的匹配符

·       *(星号)表示一个单词

·       #(井号)表示零个或者多个单词

示例说明:

这一章的例子中,我们使用三个段式的路由关键字,有三个单词和两个点组成。第一个词是速度,第二个词是颜色,第三个是动物名称。

我们用三个关键字来绑定,Q1绑定关键字是【*.orange.*】,Q2绑定关键字是【*.*.rabbit】和【lazy.#】,然后分析会发生什么:



·       Q1会收到所有orange这种颜色相关的消息

·       Q2会收到所有rabbit这个动物相关的消息和所有速度lazy的动物的消息

分析:

生产者发送“quick.orange.rabbit”的消息,两个队列都会收到

生产者发送“lazy.orange.elephant”,两队列也都会收到。

生产者发送"quick.orange.fox",那么只有Q1会收到。

生产者发送"lazy.brown.fox",那么只会有Q2能收到。

生产者发送"quick.brown.fox",那么这条消息会被丢弃,谁也收不到。

生产者发送"quick.orange.male.rabbit",这个消息也会被丢弃,谁也收不到。

生产者发送"lazy.orange.male.rabbit",这个消息会被Q2的【lazy.#】规则匹配上,发送到Q2队列中。



注意

交换器在匹配模式下:

如果消费者端的路由关键字只使用【#】来匹配消息,在匹配【topic】模式下,它会变成一个分发【fanout】模式,接收所有消息。

如果消费者端的路由关键字中没有【#】或者【*】,它就变成直连【direct】模式来工作。







测试代码

包图

代码

ReceiveLogsTopic1.java

[plain] view plain copy

1.  import com.rabbitmq.client.*;  

2.  import java.io.IOException;  

3.    

4.  public class ReceiveLogsTopic1 {  

5.    

6.      private static final String EXCHANGE_NAME = "topic_logs";  

7.         

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

9.          ConnectionFactory factory = new ConnectionFactory();  

10.         factory.setHost("localhost");  

11.         Connection connection = factory.newConnection();  

12.         Channel channel = connection.createChannel();  

13. //      声明一个匹配模式的交换器  

14.         channel.exchangeDeclare(EXCHANGE_NAME, "topic");  

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

16.         // 路由关键字  

17.         String[] routingKeys = new String[]{"*.orange.*"};  

18. //      绑定路由关键字  

19.         for (String bindingKey : routingKeys) {  

20.             channel.queueBind(queueName, EXCHANGE_NAME, bindingKey);  

21.             System.out.println("ReceiveLogsTopic1 exchange:"+EXCHANGE_NAME+", queue:"+queueName+", BindRoutingKey:" + bindingKey);  

22.         }  

23.   

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

25.   

26.         Consumer consumer = new DefaultConsumer(channel) {  

27.             @Override  

28.             public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {  

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

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

31.             }  

32.         };  

33.         channel.basicConsume(queueName, true, consumer);  

34.     }  

35. }  


ReceiveLogsTopic2.java

[plain] view plain copy

1.  import com.rabbitmq.client.*;  

2.  import java.io.IOException;  

3.    

4.  public class ReceiveLogsTopic2 {  

5.    

6.      private static final String EXCHANGE_NAME = "topic_logs";  

7.         

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

9.          ConnectionFactory factory = new ConnectionFactory();  

10.         factory.setHost("localhost");  

11.         Connection connection = factory.newConnection();  

12.         Channel channel = connection.createChannel();  

13. //      声明一个匹配模式的交换器  

14.         channel.exchangeDeclare(EXCHANGE_NAME, "topic");  

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

16.         // 路由关键字  

17.         String[] routingKeys = new String[]{"*.*.rabbit", "lazy.#"};  

18. //      绑定路由关键字  

19.         for (String bindingKey : routingKeys) {  

20.             channel.queueBind(queueName, EXCHANGE_NAME, bindingKey);  

21.             System.out.println("ReceiveLogsTopic2 exchange:"+EXCHANGE_NAME+", queue:"+queueName+", BindRoutingKey:" + bindingKey);  

22.         }  

23.   

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

25.   

26.         Consumer consumer = new DefaultConsumer(channel) {  

27.             @Override  

28.             public void handleDelivery(String consumerTag, Envelope envelope, AMQP.BasicProperties properties, byte[] body) throws IOException {  

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

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

31.             }  

32.         };  

33.         channel.basicConsume(queueName, true, consumer);  

34.     }  

35. }  


TopicSend.java

[plain] view plain copy

1.  import com.rabbitmq.client.ConnectionFactory;  

2.  import com.rabbitmq.client.Connection;  

3.  import com.rabbitmq.client.Channel;  

4.    

5.  public class TopicSend {  

6.    

7.      private static final String EXCHANGE_NAME = "topic_logs";  

8.    

9.      public static void main(String[] argv) {  

10.         Connection connection = null;  

11.         Channel channel = null;  

12.         try {  

13.             ConnectionFactory factory = new ConnectionFactory();  

14.             factory.setHost("localhost");  

15.   

16.             connection = factory.newConnection();  

17.             channel = connection.createChannel();  

18. //          声明一个匹配模式的交换器  

19.             channel.exchangeDeclare(EXCHANGE_NAME, "topic");  

20.   

21.             // 待发送的消息  

22.             String[] routingKeys = new String[]{"quick.orange.rabbit",   

23.                                                 "lazy.orange.elephant",   

24.                                                 "quick.orange.fox",   

25.                                                 "lazy.brown.fox",   

26.                                                 "quick.brown.fox",   

27.                                                 "quick.orange.male.rabbit",   

28.                                                 "lazy.orange.male.rabbit"};  

29. //          发送消息  

30.             for(String severity :routingKeys){  

31.                 String message = "From "+severity+" routingKey' s message!";  

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

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

34.             }  

35.               

36.         } catch (Exception e) {  

37.             e.printStackTrace();  

38.         } finally {  

39.             if (connection != null) {  

40.                 try {  

41.                     connection.close();  

42.                 } catch (Exception ignore) {  

43.                 }  

44.             }  

45.         }  

46.     }  

47. }  


先运行ReceiveLogsTopic1.java

[plain] view plain copy

1.  ReceiveLogsTopic1 [*] Waiting for messages. To exit press CTRL+C  




再运行ReceiveLogsTopic2

[plain] view plain copy

1.  ReceiveLogsTopic2 exchange:topic_logs, queue:amq.gen-JwqUJNUGpdFGkeY5B6TsLw, BindRoutingKey:*.*.rabbit  

2.  ReceiveLogsTopic2 exchange:topic_logs, queue:amq.gen-JwqUJNUGpdFGkeY5B6TsLw, BindRoutingKey:lazy.#  

3.  ReceiveLogsTopic2 [*] Waiting for messages. To exit press CTRL+C  


然后运行TopicSend.java发送7条消息

[plain] view plain copy

1.  TopicSend [x] Sent 'quick.orange.rabbit':'From quick.orange.rabbit routingKey' s message!'  

2.  TopicSend [x] Sent 'lazy.orange.elephant':'From lazy.orange.elephant routingKey' s message!'  

3.  TopicSend [x] Sent 'quick.orange.fox':'From quick.orange.fox routingKey' s message!'  

4.  TopicSend [x] Sent 'lazy.brown.fox':'From lazy.brown.fox routingKey' s message!'  

5.  TopicSend [x] Sent 'quick.brown.fox':'From quick.brown.fox routingKey' s message!'  

6.  TopicSend [x] Sent 'quick.orange.male.rabbit':'From quick.orange.male.rabbit routingKey' s message!'  

7.  TopicSend [x] Sent 'lazy.orange.male.rabbit':'From lazy.orange.male.rabbit routingKey' s message!'  


再看ReceiveLogsTopic1.java,收到3条匹配的消息。

[plain] view plain copy

1.  ReceiveLogsTopic1 [x] Received 'quick.orange.rabbit':'From quick.orange.rabbit routingKey' s message!'  

2.  ReceiveLogsTopic1 [x] Received 'lazy.orange.elephant':'From lazy.orange.elephant routingKey' s message!'  

3.  ReceiveLogsTopic1 [x] Received 'quick.orange.fox':'From quick.orange.fox routingKey' s message!'  


再看ReceiveLogsTopic2.java,收到4条匹配的消息。

[plain] view plain copy

1.  ReceiveLogsTopic2 [x] Received 'quick.orange.rabbit':'From quick.orange.rabbit routingKey' s message!'  

2.  ReceiveLogsTopic2 [x] Received 'lazy.orange.elephant':'From lazy.orange.elephant routingKey' s message!'  

3.  ReceiveLogsTopic2 [x] Received 'lazy.brown.fox':'From lazy.brown.fox routingKey' s message!'  

4.  ReceiveLogsTopic2 [x] Received 'lazy.orange.male.rabbit':'From lazy.orange.male.rabbit routingKey' s message!'  



最后,咱们来开动脑筋看看下面的题,当是我留的课外作业:

1.    在匹配交互器模式下,消费者端路由关键字 “*”能接收到生产者端发来路由关键字为空的消息吗?

2.    在匹配交换器模式下,消费者端路由关键字“#.*”能接收到生产者端以“..”为关键字的消息吗?如果发送来的消息只有一个单词,能匹配上吗?

3.    “a.*.#” 与 “a.#” 有什么不同吗?




 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
数字乡村和智慧农业的数字化转型是当前农业发展的新趋势,旨在通过应用数字技术,实现农业全流程的再造和全生命周期的管理服务。国政府高度重视这一领域的发展,提出“数字国”和“乡村振兴”战略,以提升国家治理能力,推动城乡融合发展。 数字乡村的建设面临乡村治理、基础设施、产业链条和公共服务等方面的问题,需要分阶段实施《数字乡村发展战略纲要》来解决。农业数字化转型的需求包括满足市民对优质农产品的需求、解决产销对接问题、形成优质优价机制、提高农业劳动力素质、打破信息孤岛、提高农业政策服务的精准度和有效性,以及解决农业融资难的问题。 数字乡村建设的关键在于构建“1+3+4+1”工程,即以新技术、新要素、新商业、新农民、新文化、新农村为核心,推进数据融合,强化农业大数据的汇集功能。数字农业大数据解决方案以农业数字底图和数据资源为基础,通过可视化监管,实现区域农业的全面数字化管理。 数字农业大数据架构基于大数据、区块链、GIS和物联网技术,构建农业大数据心、农业物联网平台和农村综合服务指挥决策平台三大基础平台。农业大数据心汇聚各类涉农信息资源和业务数据,支持大数据应用。信息采集系统覆盖市、县、乡、村多级,形成高效的农业大数据信息采集体系。 农业物联网平台包括环境监测系统、视频监控系统、预警预报系统和智能控制系统,通过收集和监测数据,实现对农业环境和生产过程的智能化管理。综合服务指挥决策平台利用数据分析和GIS技术,为农业决策提供支持。 数字乡村建设包括三大服务平台:治理服务平台、民生服务平台和产业服务平台。治理服务平台通过大数据和AI技术,实现乡村治理的数字化;民生服务平台利用互联网技术,提供各类民生服务;产业服务平台融合政企关系,支持农业产业发展。 数字乡村的应用场景广泛,包括农业生产过程、农产品流通、农业管理和农村社会服务。农业生产管理系统利用AIoT技术,实现农业生产的标准化和智能化。农产品智慧流通管理系统和溯源管理系统提高流通效率和产品追溯能力。智慧农业管理通过互联网+农业,提升农业管理的科学性和效率。农村社会服务则通过数字化手段,提高农村地区的公共服务水平。 总体而言,数字乡村和智慧农业的建设,不仅能够提升农业生产效率和管理水平,还能够促进农村地区的社会经济发展,实现城乡融合发展,是推动国农业现代化的重要途径。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值