轻松搞定RabbitMQ(五)——路由选择
转自 http://blog.csdn.net/xiaoxian8023/article/details/48733249
翻译地址:http://www.rabbitmq.com/tutorials/tutorial-four-java.html
在前篇博文中,我们建立了一个简单的日志系统。可以广播消息给多个消费者。本篇博文,我们将添加新的特性——我们可以只订阅部分消息。比如:我们可以接收Error级别的消息写入文件。同时仍然可以在控制台打印所有日志。
Bindings(绑定)
在上一篇博客中我们已经使用过绑定。类似下面的代码:
- channel.queueBind(queueName, EXCHANGE_NAME, "");
绑定表示转换器与队列之间的关系。可以简单的人为:队列对该转发器上的消息感兴趣。
绑定可以设定额外的routingKey参数。为了与避免basicPublish方法(发布消息的方法)的参数混淆,我们准备把它称作绑定键(binding key)。下面展示如何使用绑定键(binding key)来创建一个绑定:
- 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类型的转发器。我们将 提供日志严重性做为绑定键。那样,接收程序可以选择性的接收严重性的消息。首先关注发送日志的代码:
像往常一样首先创建一个转换器:
- 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);
- }
完整的例子
发送端代码(EmitLogDirect.java)
- public class EmitLogDirect {
- private final static String EXCHANGE_NAME = "direct_logs";
- public static void main(String[] args) throws IOException {
- /**
- * 创建连接连接到MabbitMQ
- */
- ConnectionFactory factory = new ConnectionFactory();
- // 设置MabbitMQ所在主机ip或者主机名
- factory.setHost("127.0.0.1");
- // 创建一个连接
- Connection connection = factory.newConnection();
- // 创建一个频道
- Channel channel = connection.createChannel();
- // 指定转发——广播
- channel.exchangeDeclare(EXCHANGE_NAME, "direct");
- //所有日志严重性级别
- String[] severities={"error","info","warning"};
- for(int i=0;i<3;i++){
- String severity = severities[i%3];//每一次发送一条不同严重性的日志
- // 发送的消息
- String message = "Hello World"+Strings.repeat(".", i+1);
- //参数1:exchange name
- //参数2:routing key
- channel.basicPublish(EXCHANGE_NAME, severity, null, message.getBytes());
- System.out.println(" [x] Sent '" + severity +"':'"+ message + "'");
- }
- // 关闭频道和连接
- channel.close();
- connection.close();
- }
- }
消费者1(ReceiveLogs2Console.java)
- public class ReceiveLogs2Console {
- private static final String EXCHANGE_NAME = "direct_logs";
- public static void main(String[] argv) throws IOException, InterruptedException {
- ConnectionFactory factory = new ConnectionFactory();
- factory.setHost("127.0.0.1");
- // 打开连接和创建频道,与发送端一样
- Connection connection = factory.newConnection();
- final Channel channel = connection.createChannel();
- channel.exchangeDeclare(EXCHANGE_NAME, "direct");
- // 声明一个随机队列
- String queueName = channel.queueDeclare().getQueue();
- //所有日志严重性级别
- String[] severities={"error","info","warning"};
- for (String severity : severities) {
- //关注所有级别的日志(多重绑定)
- channel.queueBind(queueName, EXCHANGE_NAME, severity);
- }
- System.out.println(" [*] Waiting for messages. To exit press CTRL+C");
- // 创建队列消费者
- final 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);
- }
- }
消费者2(ReceiveLogs2File.java)
- public class ReceiveLogs2File {
- private static final String EXCHANGE_NAME = "direct_logs";
- public static void main(String[] argv) throws IOException, InterruptedException {
- ConnectionFactory factory = new ConnectionFactory();
- factory.setHost("127.0.0.1");
- // 打开连接和创建频道,与发送端一样
- Connection connection = factory.newConnection();
- final Channel channel = connection.createChannel();
- channel.exchangeDeclare(EXCHANGE_NAME, "direct");
- // 声明一个随机队列
- String queueName = channel.queueDeclare().getQueue();
- String severity="error";//只关注error级别的日志,然后记录到文件中去。
- channel.queueBind(queueName, EXCHANGE_NAME, severity);
- System.out.println(" [*] Waiting for messages. To exit press CTRL+C");
- // 创建队列消费者
- final 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");
- //记录日志到文件:
- print2File( "["+ envelope.getRoutingKey() + "] "+message);
- }
- };
- channel.basicConsume(queueName, true, consumer);
- }
- private static void print2File(String msg) {
- try {
- String dir = ReceiveLogs2File.class.getClassLoader().getResource("").getPath();
- String logFileName = new SimpleDateFormat("yyyy-MM-dd").format(new Date());
- File file = new File(dir, logFileName + ".log");
- FileOutputStream fos = new FileOutputStream(file, true);
- fos.write((new SimpleDateFormat("HH:mm:ss").format(new Date())+" - "+msg + "\r\n").getBytes());
- fos.flush();
- fos.close();
- } catch (FileNotFoundException e) {
- e.printStackTrace();
- } catch (IOException e) {
- e.printStackTrace();
- }
- }
- }
最终结果:
轻松搞定RabbitMQ(五)——路由选择的更多相关文章
- 轻松搞定RabbitMQ(六)——主题
转自 http://blog.csdn.net/xiaoxian8023/article/details/48806871 翻译地址:http://www.rabbitmq.com/tutorials ...
- 轻松搞定RabbitMQ(二)——工作队列之消息分发机制
转自 http://blog.csdn.net/xiaoxian8023/article/details/48681987 上一篇博文中简单介绍了一下RabbitMQ的基础知识,并写了一个经典语言入门 ...
- 轻松搞定RabbitMQ(一)——RabbitMQ基础知识+HelloWorld
转自 http://blog.csdn.net/xiaoxian8023/article/details/48679609 本文是简单介绍一下RabbitMQ,参考官网上的教程.同时加入了一些自己的理 ...
- 轻松搞定RabbitMQ(四)——发布/订阅
转自 http://blog.csdn.net/xiaoxian8023/article/details/48729479 翻译地址:http://www.rabbitmq.com/tutorials ...
- 轻松搞定RabbitMQ(三)——消息应答与消息持久化
转自 http://blog.csdn.net/xiaoxian8023/article/details/48710653 这个官网的第二个例子中的消息应答和消息持久化部分.我把它摘出来作为单独的一块 ...
- 【转】轻松搞定FTP之FlashFxp全攻略
转载网址:http://www.newhua.com/2008/0603/39163.shtml 轻松搞定FTP之FlashFxp全攻略 导读: FlashFXP是一款功能强大的FXP/FTP软件,融 ...
- Webcast / 技术小视频制作方法——自己动手录制video轻松搞定
Webcast / 技术小视频制作方法——自己动手录制video轻松搞定 http://blog.sina.com.cn/s/blog_67d387490100wdnh.html 最近申请加入MSP的 ...
- 【微服务】之五:轻松搞定SpringCloud微服务-调用远程组件Feign
上一篇文章讲到了负载均衡在Spring Cloud体系中的体现,其实Spring Cloud是提供了多种客户端调用的组件,各个微服务都是以HTTP接口的形式暴露自身服务的,因此在调用远程服务时就必须使 ...
- 使用BleLib的轻松搞定Android低功耗蓝牙Ble 4.0开发具体解释
转载请注明来源: http://blog.csdn.net/kjunchen/article/details/50909410 使用BleLib的轻松搞定Android低功耗蓝牙Ble 4.0开发具体 ...
随机推荐
- SpriteKit-SKView
1.暂停这个视图 @property (nonatomic, getter = isPaused) BOOL paused; 2.视图性能的一些参数 @property (nonatomic) BOO ...
- ubuntu 安装tomcat<服务器>
一.下载tomcat 可以先下载到本地,然后ftp到服务器 官方 Apache Tomcat 的下载页面(下面的链接是apache自己的镜像服务器的地址,不同网络连接的话,apache会给出不同的镜像 ...
- Problem 1004: 蛤玮打扫教室(区间覆盖端点记录)
Problem 1004: 蛤玮打扫教室 Time Limits: 1000 MS Memory Limits: 65536 KB 64-bit interger IO format: %l ...
- ACM程序设计选修课——Problem F:(ds:图)旅游规划(优先队列+SPFA)
问题 F: (ds:图)旅游规划 时间限制: 1 Sec 内存限制: 128 MB 提交: 14 解决: 4 题目描述 有了一张自驾旅游路线图,你会知道城市间的高速公路长度.以及该公路要收取的过路 ...
- P2134 百日旅行 (斜率优化,DP)
题目链接 Solution 斜率优化\(DP\). 今天下午才打的第一道题 QwQ... \(90\) 分很简单,一个简单的递推. 令 \(f[i]\) 为最后一天旅游的花费, \(g[i]\) 为最 ...
- 旅行问题(bzoj 2746)
Description yz是Z国的领导人,他规定每个地区的名字只能为26个小写拉丁字母的一个.由于地 区数有可能超过26个,便产生了一个问题,如何辨别名字相同的地区?于是yz规定,一个 地区的描述必 ...
- 【CF1015A】Points in Segments(签到)
题意:有一条上面有n个点的数轴,给定m次操作,每次覆盖(x[i],y[i]),求最后没有被覆盖过的点的数量与他们的编号 n,m<=100 思路: #include<cstdio> # ...
- 标准C程序设计七---101
Linux应用 编程深入 语言编程 标准C程序设计七---经典C11程序设计 以下内容为阅读: <标准C程序设计>(第7版) 作者 ...
- Scrapy学习-13-使用DownloaderMiddleware设置IP代理池及IP变换
设置IP代理池及IP变换方案 方案一: 使用国内免费的IP代理 http://www.xicidaili.com # 创建一个tools文件夹,新建一个py文件,用于获取代理IP和PORT from ...
- cookies-判断用户是否是第一次进入页面
在郑州美莱的活动项目中客户当时有提到该需求.尽管最后去掉了该需求,但是还是花了我不少时间研究,因为之前的项目我前端没有用到过cookies,吓死宝宝了 $(document).ready(functi ...