如何将普通队列异常消息路由到死信队列

如何将普通队列异常消息路由到死信队列

在RabbitMQ中,我们经常需要处理异常消息,特别是当消息处理失败时。一个常见的用例是将异常消息路由到死信队列,以便稍后重新处理或进一步分析。在本篇博客中,我们将演示如何配置RabbitMQ和Spring Boot,以便将普通队列中的异常消息自动路由到死信队列。

步骤1:创建Spring Boot项目

首先,创建一个Spring Boot项目并添加以下依赖:

xml 复制代码
<dependencies>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-amqp</artifactId>
    </dependency>
    <dependency>
        <groupId>cn.hutool</groupId>
        <artifactId>hutool-all</artifactId>
        <version>5.8.0</version>
    </dependency>
</dependencies>

步骤2:配置RabbitMQ和队列

在`application.yml文件中,配置RabbitMQ连接信息:

properties 复制代码
spring:
  #给项目来个名字
  application:
    name: rabbitmq-provider
  #配置rabbitMq 服务器
  rabbitmq:
    host: 127.0.0.1
    port: 5672
    username: root
    password: 123456
    #虚拟host 可以不设置,使用server默认host
    virtual-host: /cdn
      #确认消息已发送到交换机(Exchange)
      #确认消息已发送到队列(Queue)
    publisher-returns: true
    listener:
      simple:
        acknowledge-mode: auto
        retry:
          #enabled:开启失败重试
          enabled: true
          #第一次重试的间隔时长
          initial-interval: 1000ms
          #最长重试间隔,超过这个间隔将不再重试
          max-interval: 300000ms
          #下次重试间隔的倍数,此处是2即下次重试间隔是上次的2倍
          multiplier: 2
          max-attempts: 4
        default-requeue-rejected: false  # 达到重试次数进入死信队列
    publisher-confirm-type: none

然后,创建一个Spring组件,它将包括配置RabbitMQ队列和消息监听器:

java 复制代码
import org.springframework.amqp.core.*;
import org.springframework.amqp.rabbit.annotation.RabbitListener;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.stereotype.Component;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
import cn.hutool.core.util.RandomUtil;

@RestController
public class ExceptionToDead {

    @Autowired
    RabbitTemplate rabbitTemplate;

    @Bean
    public Queue normalQueue() {
        return QueueBuilder.durable("my-normal-queue")
                .withArgument("x-dead-letter-exchange", "my-dead-letter-exchange")
                .withArgument("x-dead-letter-routing-key", "my-dead-letter-routing-key")
                .build();
    }

    @Bean
    public DirectExchange normalExchange() {
        return new DirectExchange("my-normal-exchange");
    }

    @Bean
    public Binding normalBinding(Queue normalQueue, DirectExchange normalExchange) {
        return BindingBuilder.bind(normalQueue).to(normalExchange).with("my-normal-routing-key");
    }

    @Bean
    public DirectExchange deadLetterExchange() {
        return new DirectExchange("my-dead-letter-exchange");
    }

    @Bean
    public Queue deadLetterQueue() {
        return QueueBuilder.durable("my-dead-letter-queue").build();
    }

    @Bean
    public Binding deadLetterBinding(Queue deadLetterQueue, DirectExchange deadLetterExchange) {
        return BindingBuilder.bind(deadLetterQueue).to(deadLetterExchange).with("my-dead-letter-routing-key");
    }

    @RabbitListener(queues = "my-normal-queue")
    public void receiveFromNormalQueue(Integer message) {
        System.out.println("正常队列收到消息: " + message);
        if (message % 2 == 0) {
            int a = 1 / 0; // 引发异常
        }
    }

    @RabbitListener(queues = "my-dead-letter-queue")
    public void receiveFromDeadLetterQueue(Integer message) {
        System.out.println("死信队列收到消息: " + message);
    }

    @GetMapping("send")
    public void send() {
        int number = RandomUtil.randomInt(0, 100);
        rabbitTemplate.convertAndSend("my-normal-exchange", "my-normal-routing-key", number);
    }

    @GetMapping("b")
    public void b() {
        int number = RandomUtil.randomInt(0, 100);
        rabbitTemplate.convertAndSend("simple-key", number);
    }
}

在这段代码中,我们创建了一个名为my-normal-queue的队列,它将异常消息路由到my-dead-letter-exchange交换机,然后再路由到my-dead-letter-queue队列。我们还创建了相应的交换机和绑定。

注意: 自动确认的时候 ,达到重试次数之后就会路由到绑定的死信,否则消息会丢失

​ 手动确认的时候,消费时候在需要的位置channel.basicNack(message.getMessageProperties().getDeliveryTag(),false,false); 消息才会路由到死信,否则消息会积压

步骤3:监听异常消息并触发死信路由

在上述代码中,我们使用@RabbitListener注解来监听my-normal-queue队列。当消息处理失败时(在这里,我们模拟了一个条件,如果消息为偶数,则引发异常),它将被路由到死信队列。

步骤4:触发异常消息

最后,我们在/send端点上创建了一个HTTP请求,它将发送一个随机整数到my-normal-queue队列。如果随机整数

相关推荐
NiNg_1_2343 小时前
SpringBoot整合SpringSecurity实现密码加密解密、登录认证退出功能
java·spring boot·后端
种树人202408193 小时前
如何在 Spring Boot 中启用定时任务
spring boot
Chrikk4 小时前
Go-性能调优实战案例
开发语言·后端·golang
幼儿园老大*4 小时前
Go的环境搭建以及GoLand安装教程
开发语言·经验分享·后端·golang·go
canyuemanyue4 小时前
go语言连续监控事件并回调处理
开发语言·后端·golang
杜杜的man4 小时前
【go从零单排】go语言中的指针
开发语言·后端·golang
苹果醋36 小时前
Java8->Java19的初步探索
java·运维·spring boot·mysql·nginx
Wx-bishekaifayuan6 小时前
django电商易购系统-计算机设计毕业源码61059
java·spring boot·spring·spring cloud·django·sqlite·guava
customer086 小时前
【开源免费】基于SpringBoot+Vue.JS周边产品销售网站(JAVA毕业设计)
java·vue.js·spring boot·后端·spring cloud·java-ee·开源