Rabbitmq Ack 或 Nack,将消息留在队列中
Rabbitmq Ack or Nack, leaving messages on the queue
我一直在研究 RabbitMq.net 和消息确认。
如果消费者能够处理消息,您可以以
的形式发回确认
channel.BasicAck(ea.DeliveryTag, false);
这会将其从队列中移除。
但是如果消息无法处理怎么办?可能是临时中断,您不希望从队列中取出的消息只是放在后面继续处理下一条消息?
我试过使用
channel.BasicNack(ea.DeliveryTag, false, true);
但下一次它仍然收到相同的消息,而不是移动到队列中的下一条消息
我的完整代码是
class Program
{
private static IModel channel;
private static QueueingBasicConsumer consumer;
private static IConnection Connection;
static void Main(string[] args)
{
Connection = GetRabbitMqConnection();
channel = Connection.CreateModel();
channel.BasicQos(0, 1, false);
consumer = new QueueingBasicConsumer(channel);
channel.BasicConsume("SMSQueue", false, consumer);
while (true)
{
if (!channel.IsOpen)
{
throw new Exception("Channel is closed");
}
var ea = consumer.Queue.Dequeue();
string jsonified = Encoding.UTF8.GetString(ea.Body);
var message = JsonConvert.DeserializeObject<SmsRecords>(jsonified);
if (ProcessMessage())
channel.BasicAck(ea.DeliveryTag, false);
else
channel.BasicNack(ea.DeliveryTag, false, true);
}
}
private static bool ProcessMessage()
{
return false;
}
public static IConnection GetRabbitMqConnection()
{
try
{
var connectionFactory = new ConnectionFactory
{
UserName = "guest",
Password = "guest",
HostName = "localhost"
};
return connectionFactory.CreateConnection();
}
catch (Exception ex)
{
Console.WriteLine(ex.Message);
return null;
}
}
}
我的公司是这样做的:如果一条消息失败(出于任何原因),我们会将消息放入一个保留队列中等待 10 秒钟,然后将其放回队列中以进行重试。我们最多执行此循环 10 次,如果消息被 nacked 10 次,那么我们认为这是一个我们无法恢复的故障,我们将其放入永久死信队列以进行调查。
这是示意图:
我一直在研究 RabbitMq.net 和消息确认。 如果消费者能够处理消息,您可以以
的形式发回确认channel.BasicAck(ea.DeliveryTag, false);
这会将其从队列中移除。
但是如果消息无法处理怎么办?可能是临时中断,您不希望从队列中取出的消息只是放在后面继续处理下一条消息?
我试过使用
channel.BasicNack(ea.DeliveryTag, false, true);
但下一次它仍然收到相同的消息,而不是移动到队列中的下一条消息
我的完整代码是
class Program
{
private static IModel channel;
private static QueueingBasicConsumer consumer;
private static IConnection Connection;
static void Main(string[] args)
{
Connection = GetRabbitMqConnection();
channel = Connection.CreateModel();
channel.BasicQos(0, 1, false);
consumer = new QueueingBasicConsumer(channel);
channel.BasicConsume("SMSQueue", false, consumer);
while (true)
{
if (!channel.IsOpen)
{
throw new Exception("Channel is closed");
}
var ea = consumer.Queue.Dequeue();
string jsonified = Encoding.UTF8.GetString(ea.Body);
var message = JsonConvert.DeserializeObject<SmsRecords>(jsonified);
if (ProcessMessage())
channel.BasicAck(ea.DeliveryTag, false);
else
channel.BasicNack(ea.DeliveryTag, false, true);
}
}
private static bool ProcessMessage()
{
return false;
}
public static IConnection GetRabbitMqConnection()
{
try
{
var connectionFactory = new ConnectionFactory
{
UserName = "guest",
Password = "guest",
HostName = "localhost"
};
return connectionFactory.CreateConnection();
}
catch (Exception ex)
{
Console.WriteLine(ex.Message);
return null;
}
}
}
我的公司是这样做的:如果一条消息失败(出于任何原因),我们会将消息放入一个保留队列中等待 10 秒钟,然后将其放回队列中以进行重试。我们最多执行此循环 10 次,如果消息被 nacked 10 次,那么我们认为这是一个我们无法恢复的故障,我们将其放入永久死信队列以进行调查。
这是示意图: