.Net RabbitMQ实战指南——进阶(一)
备份交换器
备份交换器,英文名称为Alternate Exchange,简称AE。通过在声明交换器(调用channel.ExchangeDeclare方法)时添加alternate-exchange参数来实现。
备份交换器工作流程如下:

using (var channel = connection.CreateModel())
{
//设置备胎交换器参数
var arguments = new Dictionary<string, object>();
arguments.Add("alternate-exchange","myAe");
channel.ExchangeDeclare("normalExchange", "direct",true,false, arguments);
channel.ExchangeDeclare("myAe", "fanout", true, false);
channel.QueueDeclare("nromalQueue",true,false,false);
channel.QueueBind("nromalQueue", "normalExchange", "normalKey");
channel.QueueDeclare("unroutedQueue", true, false, false);
channel.QueueBind("unroutedQueue", "myAe","ae"); var properties = channel.CreateBasicProperties();
properties.DeliveryMode = 2;
string message = "RabbitMQ Test"; //传递的消息内容
channel.BasicPublish("normalExchange", "normalKey", properties, Encoding.UTF8.GetBytes(message)); //生产消息
channel.BasicPublish("normalExchange", "un-routkey", properties, Encoding.UTF8.GetBytes(message)); //生产消息 Console.WriteLine($"Send:{message}");
}
代码中声明了两个交换器normalExchange和myAe,分别绑定了normalQueue和unroutedQueue这两个队列,同时将myAe设置为normalExchange的备份交换器。myAe的交换器类型为fanout。同时生成两条消息,其中“un-routkey”并没有定义对应路由。
运行效果:

切换到queues视图,可以看到两个队列分别有一条需要消费的消息。

备份交换器与普通的交换器没有太大的区别,为了方便使用,建议设置为fanout类型
过期时间(TTL)
TTL,Time to Live的简称,即过期时间。RabbitMQ可以对队列和消息设置TTL。
设置队列的TTL
channel.QueueDeclare方法中的x-expires参数可以设置队列被自动删除前处于未使用状态的时间。未使用是指队列上没有任何的消费者,队列也没有被重新声明过,并且其间内也未调用过Basic.Get命令。
var arguments = new Dictionary<string, object>();
arguments.Add("x-expires", 10000); //单位毫秒
channel.QueueDeclare(arguments: arguments);
RabbitMQ重启后,持久化的队列的过期时间会被重新计算。
设置消息的TTL
消息的过期时间有两种设置方式。第一种是通过队列属性设置,队列中所有消息都有相同的过期时间。第二种方法是对消息本身进行设置。
如果两种方法一起使用,则消息的TTL以两者之间较小的那个数值为准。消息在队列中的生存时间一旦超过设置的TTL值,就会变成“死信”(Dead Message)
通过队列属性设置消息TTL的方法是在channel.QueueDeclare方法中加入x-message-ttl参数实现的,这个参数的单位是毫秒。
arguments.Add("x-message-ttl", 6000); //单位毫秒
channel.QueueDeclare(arguments: arguments);
对消息本身进行过期时间设置前面代码有涉及过,是通过BasicPublish方法的basicProperties参数指定。
var properties = channel.CreateBasicProperties();
properties.DeliveryMode = 1;
properties.Priority = 2;
properties.Expiration = "6000";
var message = "RabbitMQ Test"; //传递的消息内容
channel.BasicPublish("", "stacking", properties, Encoding.UTF8.GetBytes(message)); //生产消息
死信队列
DLX,全称为Dead-Letter-Exchange,即死信交换器。当一个消息在队列中变成死信(dead message),它就会被重新被发送到DLX(死信交换器),绑定DLX的队列就称之为死信队列。
通过在channel.QueueDeclare方法中设置x-dead-letter-exchange参数来为这个队列添加DLX.
示例代码:
var arguments = new Dictionary<string, object>();
arguments.Add("x-expires", 10000); //单位毫秒
arguments.Add("x-message-ttl", 6000); //单位毫秒
channel.QueueDeclare(arguments: arguments); channel.ExchangeDeclare("dlx_exchange", "direct");
var argumentsDlx = new Dictionary<string, object>();
argumentsDlx.Add("x-dead-letter-exchange", "dlx_exchange");
argumentsDlx.Add("x-dead-letter-routing-key", "dlx-routing-key"); //为DLX指定路由键,如果没有特殊指定,则使用原队列的路由键
channel.QueueDeclare("dlx_queue",false,false,false,argumentsDlx);
可以看到dlx_queue有标记了DLX和DLK(Dead-Letter-Key)。

DLX是一个非常有用的特性。异常情况下,消息不能够被消费者正确消费(消费者调用了Basic.Nack或者Basic.Reject)而被置入死信队列中,后续分析程序可以通过死信队列中的内容来分析当时所遇到的异常情况,从而改善和优化系统。
延迟队列
延迟队列存储的是对应的延迟消息,“延迟消息”并不想让消费者立刻消息,而是等待特定时间后,消费者才能拿到这个消息进行消费。如一个订单系统中,用户下单30分钟内没有完成支付那么就对该订单进行异常处理,就可以使用延迟队列来处理这些订单。
RabbitMQ并支持延迟队列的功能,但是我们可以通过DLX和TTL模拟出延迟队列的功能。如下图所示:

实例中设置了5秒、10秒、30秒、1分钟四个延时等级。根据应用需求的不同,生产者在发送消息的时候设置不同的路由键,从而将消息发送到与交换器绑定的对应队列中。这些队列分别设置了过期时间为5秒、10秒、30秒、1分钟,同时也分别配置了DLX和相应的死信队列。队列中消息过期时,就会转存到相应的死信队列(即延迟队列)中,在根据业务自身的情况,分别选择不同延迟等级的延迟队列进行消费。
优先级队列
优先级队列,有高优先级的队列具有高的优先权,优先级高的消息有优先被消费的特权。
可以通过设置队列的x-max-priority参数来实现。并在发送消息时通过Priority设置消息优先级。
using (var channel = connection.CreateModel())
{ channel.ExchangeDeclare("priority_exchange", "direct");
var arguments = new Dictionary<string, object>();
arguments.Add("x-max-priority", 10); //
channel.QueueDeclare("priority_queue", false, false, false, arguments);
channel.QueueBind("priority_queue", "priority_exchange", "priority_key"); var properties = channel.CreateBasicProperties();
properties.Priority = 2;
string message = "Priority 2"; //传递的消息内容
channel.BasicPublish("priority_exchange", "priority_key", properties, Encoding.UTF8.GetBytes(message)); //生产消息
properties.Priority = 5;
message = "Priority 5"; //传递的消息内容
channel.BasicPublish("priority_exchange", "priority_key", properties, Encoding.UTF8.GetBytes(message)); //生产消息
var result = channel.BasicGet("priority_queue", false);
channel.BasicAck(result.DeliveryTag, true);
Console.WriteLine($"Received:{Encoding.UTF8.GetString(result.Body.ToArray())}"); }
运行上面的代码,不管我们如何调整生成消息的顺序,channel.BasicGet取出来的始终是Priority为5的那条消息。

Github
示例代码地址:https://github.com/MayueCif/RabbitMQ
.Net RabbitMQ实战指南——进阶(一)的更多相关文章
- .Net RabbitMQ实战指南——进阶(二)
持久化 持久化可以提高RabbitMQ的可靠性,防止异常情况下的数据丢失.RabbitMQ的持久化分为三个部分:交换器的持久化.队列的持久化和消息的持久化. 交换器的持久化通过声明队列时将durabl ...
- 【RabbitMQ 实战指南】一 RabbitMQ 开发
1.RabbitMQ 安装 RabbitMQ 的安装可以参考官方文档:https://www.rabbitmq.com/download.html 2.管理页面 rabbitmq-management ...
- 【RabbitMQ 实战指南】一 延迟队列
1.什么是延迟队列 延迟队列中存储延迟消息,延迟消息是指当消息被发送到队列中不会立即消费,而是等待一段时间后再消费该消息. 延迟队列很多应用场景,一个典型的应用场景是订单未支付超时取消,用户下单之后3 ...
- 【RabbitMQ 实战指南】一 死信队列
1.死信队列 DLX,全称为 Dead-Letter-Exchange,可以称之为死信交换器.当消息在一个队列中变成死信(dead message)之后,它能被发送到另一个交换器中,这个交换器就是DL ...
- 【RabbitMQ 实战指南】一 过期时间TTL
RabbitMQ 可以对消息和队列设置过期时间(TTL) 1.设置消息的TTL 目前有两种方式可以设置消息的TTL 第一种方式是通过队列属性设置,队列中所有消息都有相同的过期时间 第二种方式是对消息本 ...
- 【RabbitMQ 实战指南】一 RabbitMQ入门
1.消息中间件 1.1.什么是消息中间件 消息中间件(Message Queue Middleware,简称 MQ)是指利用高效可靠的消息传递机制进行与平台无关的数据交流,并基于数据通道来进行分布式系 ...
- .Net RabbitMQ实战指南——客户端开发
开发中关键的Class和Interface有Channel.Connection.ConnectionFactory.Consumer等,与RabbitMQ相关的开发工作,基本上是围绕Connecti ...
- .Net RabbitMQ实战指南——RabbitMQ相关概念介绍
什么是消息中间件 消息(Message)是指在应用间传送的数据.消息可以非常简单,比如只包含文本字符串.JSON等,也可以很复杂,比如内嵌对象. 消息队列中间件(Message Queue Middl ...
- .Net RabbitMQ实战指南——HTTP API接口调用
RabbitMQ Management插件还提供了基于RESTful风格的HTTP API接口来方便调用.一共涉及4种HTTP方法:GET.PUT.DELETE和POST.GET方法一般用来获取如集群 ...
随机推荐
- hdu3449 有依赖的背包问题
题意: 给你一些物品,每个物品有自己的价值和花费,每个物品都对应一个箱子,每个箱子有价钱,买这个物品必须买相应的箱子,给你一个价钱,问最多可以获得多少价值 <提示:多个物品可能同时对 ...
- hdu 2058 枚举区间和个数
题意: 给你两个数n,m,意思是有一个序列长度n,他是1 2 3 4 ...n,然后让你输出所有连续和等于m的范围. 思路: 是个小水题,随便写几个数字就能发现规律了,我们可以 ...
- 【python】Leetcode每日一题-颠倒二进制位
[python]Leetcode每日一题-颠倒二进制位 [题目描述] 颠倒给定的 32 位无符号整数的二进制位. 示例1: 输入: 00000010100101000001111010011100 输 ...
- .NET Core with 微服务 - 什么是微服务
微服务是这几年最流行的架构,说起架构不提微服务都不好意思跟人家打招呼.最近想要再梳理一下关于微服务的知识,并且结合本人的一些实践经验来做一些总结与分享.前面会分享一些概念性的东西,后面也会使用.net ...
- Markdown编辑器怎么用
Markdown编辑器怎么用 1.代码块 快速创建一个代码块 // 语法: // ```+语言名称,如```java,```c++ 2.标题 语法:#+空格+标题名字,一个#表示一级标题,两个#表示二 ...
- Mybatis学习之自定义持久层框架(三) 自定义持久层框架:读取并解析配置文件
前言 前两篇文章分别讲解了JDBC和Mybatis的基本知识,以及自定义持久层框架的设计思路,从这篇文章开始,我们正式来实现一个持久层框架. 新建一个项目 首先我们新建一个maven项目,将其命名为I ...
- OO第三单元总结——JML规格
一.JML简介 1.JML语言的理论基础 JML(Java Modeling Language)是用于对Java程序进行规格化设计的一种表示语言.JML是一种行为接口规格语言 (Behavior In ...
- OO第1.2次作业·魔鬼的三角函数化简
多年以后,面对办公室的屏幕,我会回忆起开始肝第二周OO作业的那个遥远的下午.那时的程序是一个一两百行的符号求导,基类与接口在包里一字排开,工整的注释一望到底 谁能想到,接下来的十几个小时我要经历什么样 ...
- linux如何patch打补丁
1. 创建2个文件 1.txt 和 2.txt 并在1.txt基础上修改成为2.txt book@100ask:~/patch$ ls 1.txt 2.txt book@100ask:~/patch$ ...
- spring为何要注入接口,而注入接口的实现类就会报错
首先说明,注入的对象确实为实现类的对象.(并不是实现类的代理对象,注入并不涉及代理) 如果只是单纯注入是可以用实现类接收注入对象的,但是往往开发中会对实现类做增强,如事务,日志等,实现增强的AOP技术 ...