목차
환경 구성
消息丢失分析
生产阶段
生产端模拟消息丢失
RabbitMQ
消费端
Java java지도 시간 SpringBoot+RabbitMQ를 사용하여 안정적인 메시지 전송을 달성하는 방법

SpringBoot+RabbitMQ를 사용하여 안정적인 메시지 전송을 달성하는 방법

May 29, 2023 pm 10:34 PM
springboot rabbitmq

    환경 구성

    SpringBootRabbitMQ를 통합하여 메시지 전송을 구현합니다. SpringBoot 整合 RabbitMQ 实现消息的发送。

    1.添加 maven 依赖

           <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter</artifactId>
            </dependency>
    
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-web</artifactId>
            </dependency>
    
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-amqp</artifactId>
            </dependency>
    로그인 후 복사

    2.添加 application.yml 配置文件

    spring:
      rabbitmq:
        host: 192.168.3.19
        port: 5672
        username: admin
        password: xxxx
    로그인 후 복사

    3.配置交换机、队列以及绑定

        @Bean
        public DirectExchange myExchange() {
            DirectExchange directExchange = new DirectExchange("myExchange");
            return directExchange;
        }
    
        @Bean
        public Queue myQueue() {
            Queue queue = new Queue("myQueue");
            return queue;
        }
    
        @Bean
        public Binding binding() {
            return BindingBuilder.bind(myQueue()).to(myExchange()).with("myRoutingKey");
        }
    로그인 후 복사

    4.生产发送消息

        @Autowired
        private RabbitTemplate rabbitTemplate;
    
        @GetMapping("/send")
        public String send(String message) {
            rabbitTemplate.convertAndSend("myExchange","myRoutingKey",message);
            System.out.println("【发送消息】" + message)
            return "【send message】" + message;
        }
    로그인 후 복사

    5.消费者接收消息

        @RabbitListener(queuesToDeclare = @Queue("myQueue"))
        public void process(String msg, Channel channel, Message message) {
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
            Date date = new Date();
            String time = sdf.format(date);
            System.out.println("【接收信息】" + msg + " 当前时间" + time);
    로그인 후 복사

    6.调用生产端发送消息 hello,控制台输出:

    【发送消息】hello
    【接收信息】hello 当前时间2022-05-12 10:21:14

    说明消息已经被成功接收。

    消息丢失分析

    SpringBoot+RabbitMQ를 사용하여 안정적인 메시지 전송을 달성하는 방법

    一条消息的从生产到消费,消息丢失可能发生在以下几个阶段:

    • 生产端丢失: 生产者无法传输到 RabbitMQ

    • 存储端丢失: RabbitMQ 存储自身挂了

    • 消费端丢失:存储由于网络问题,无法发送到消费端,或者消费挂了,无法发送正常消费

    RabbitMQ 从生产端、储存端、消费端都对可靠性传输做很好的支持。

    生产阶段

    生产阶段通过请求确认机制,来确保消息的可靠传输。当发送消息到 RabbitMQ 服务器 之后,RabbitMQ 收到消息之后,给发送返回一个请求确认,表示RabbitMQ 服务器已成功的接收到了消息。

    配置application.yml

    spring:
      rabbitmq:
        # 消息确认机制 生产者 -> 交换机
        publisher-confirms: true
        # 消息返回机制  交换机 -> 队列
        publisher-returns: true
    로그인 후 복사

    配置

    @Configuration
    @Slf4j
    public class RabbitConfig {
    
        @Autowired
        private ConnectionFactory connectionFactory;
    
        @Bean
        public RabbitTemplate rabbitTemplate() {
            RabbitTemplate rabbitTemplate = new RabbitTemplate(connectionFactory);
            rabbitTemplate.setConfirmCallback(new RabbitTemplate.ConfirmCallback() {
                @Override
                public void confirm(CorrelationData correlationData, boolean ack, String cause) {
                    log.info("【correlationData】:" + correlationData);
                    log.info("【ack】" + ack);
                    log.info("【cause】" + cause);
                    if (ack) {
                        log.info("【发送成功】");
                    } else {
                        log.info("【发送失败】correlationData:" + correlationData + " cause:" + cause);
                    }
                }
            });
            rabbitTemplate.setMandatory(true);
            rabbitTemplate.setReturnCallback(new RabbitTemplate.ReturnCallback() {
                @Override
                public void returnedMessage(Message message, int replyCode, String replyText, String exchange, String routingKey) {
                    log.warn("【消息发送失败】");
                    log.info("【message】" + message);
                    log.info("【replyCode】" + replyCode);
                }
            });
    
            return rabbitTemplate;
        }
    }
    로그인 후 복사

    消息从 生产者交换机, 有confirmCallback 确认模式。发送消息成功后消息会调用方法confirm(CorrelationData correlationData, boolean ack, String cause),根据 ack 判断消息是否发送成功。

    消息从 交换机队列,有returnCallback 退回模式。

    发送消息 product message 控制台输出如下:

    【发送消息】product message
    【接收信息】product message 当前时间2022-05-12 11:27:56
    【correlationData】:null
    【ack】true
    【cause】null
    【发送成功】

    生产端模拟消息丢失

    这里有两个方案:

    • 发送消息后立马关闭 broke,后者把网络关闭,但是broker关闭之后控制台一直就会报错,发送消息也报500错误。

    • 发送不存在的交换机:

    // myExchange 修改成 myExchangexxxxx
    rabbitTemplate.convertAndSend("myExchangexxxxx","myRoutingKey",message);
    로그인 후 복사

    结果:

    【correlationData】:null
    【ack】false
    【cause】channel error; protocol method: #method(reply-code=404, reply-text=NOT_FOUND - no exchange 'myExchangexxxxx' in vhost '/', class-id=60, method-id=40)
    【发送失败】

    当发送失败可以对消息进行重试

    交换机正确,发送不存在的队列:

    交换机接收到消息,返回成功通知,控制台输出:

    【correlationData】:CorrelationData [id=7d468b47-b422-4523-b2a2-06b14aef073c]
    【ack】true
    【cause】null
    【发送成功】

    交换机没有找到队列,返回失败信息:

    【消息发送失败】
    【message】product message
    【replyCode】312

    RabbitMQ

    开启队列持久化,创建的队列和交换机默认配置是持久化的。首先把队列和交换机设置正确,修改消费监听的队列,使得消息存放在队列里

    修改队列的持久化,修改成非持久化:

        @Bean
        public Queue myQueue() {
            Queue queue = new Queue("myQueue",false);
            return queue;
        }
    로그인 후 복사

    发送消息之后,消息存放在队列中,然后重启 RabbitMQ,消息不存在了。
    设置队列持久化:

        @Bean
        public Queue myQueue() {
            Queue queue = new Queue("myQueue",true);
            return queue;
        }
    로그인 후 복사

    重启之后,队列的消息还存在。

    消费端

    消费端默认开始 ack

    1. maven 종속성 추가

    spring:
      rabbitmq:
        # 手动消息确认
        listener:
          simple:
            acknowledge-mode: manual
    로그인 후 복사

    2. application.yml 구성 파일 추가

    channel.basicAck(message.getMessageProperties().getDeliveryTag(),false);
    로그인 후 복사
    로그인 후 복사

    3. 스위치, 대기열 및 바인딩 구성

        @RabbitListener(queuesToDeclare = @Queue("myQueue"))
        public void process(String msg, Channel channel, Message message) {
            SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
            Date date = new Date();
            String time = sdf.format(date);
            System.out.println("【接收信息】" + msg + " 当前时间" + time);
            System.out.println(message.getMessageProperties().getDeliveryTag());
            try {
                channel.basicAck(message.getMessageProperties().getDeliveryTag(),false);
            } catch (IOException e) {
                e.printStackTrace();
            }
    
        }
    로그인 후 복사

    4. 소비자는

    channel.basicAck(message.getMessageProperties().getDeliveryTag(),false);
    로그인 후 복사
    로그인 후 복사
    6라는 메시지를 받습니다. hello 메시지를 보내기 위해 프로덕션 종료를 호출합니다. 콘솔 출력은

    [Send message] hello
    [Receive message] hello입니다. 현재 시간 2022- 05-12 10:21:14

    SpringBoot+RabbitMQ를 사용하여 안정적인 메시지 전송을 달성하는 방법 메시지가 성공적으로 수신되었음을 나타냅니다.

    메시지 손실 분석

    🎜SpringBoot+RabbitMQ를 사용하여 안정적인 메시지 전송을 달성하는 방법 🎜🎜메시지 생성부터 소비까지 다음 단계에서 메시지 손실이 발생할 수 있습니다. 🎜
    • 🎜생산 종료 손실: 생산자는 RabbitMQ🎜
    • 🎜저장소 측 손실: RabbitMQ 저장소 자체가 다운되었습니다🎜
    • 🎜소비자 측 손실: 소비자 측에 네트워크 문제로 스토리지를 보낼 수 없거나 소비가 중단되어 정상적인 소비가 전송되지 않습니다🎜
    • 🎜🎜RabbitMQ는 생산 측에서 안정적인 전송을 잘 지원합니다. 저장 끝, 소비자 끝. 🎜🎜제작 단계🎜🎜제작 단계에서는 안정적인 메시지 전송을 보장하기 위해 요청 확인 메커니즘을 사용합니다. RabbitMQ 서버에 메시지를 보낸 후 RabbitMQ는 메시지를 수신하고 요청 확인을 보낸 사람에게 반환하여 RabbitMQ 서버가 메시지를 성공적으로 수신했음을 나타냅니다. 🎜🎜application.yml🎜rrreee🎜구성 🎜rrreee🎜 confirmCallback 확인 모드를 사용하여 Producer에서 switch로의 메시지를 구성합니다. 메시지가 성공적으로 전송된 후 메시지는 confirm(CorrelationData CorrelationData, boolean ack, String cause) 메서드를 호출하고 ack를 기반으로 메시지가 성공적으로 전송되었는지 확인합니다. . 🎜🎜스위치에서 로의 메시지에는 returnCallback 반환 모드가 있습니다. 🎜🎜메시지 보내기 제품 메시지 콘솔 출력은 다음과 같습니다. 🎜
      🎜【메시지 보내기】제품 메시지
      【메시지 받기】제품 메시지 현재 시간 2022-05-12 11: 27: 56
      [correlationData]:null
      [ack]true
      [cause]null
      [성공적으로 전송됨]🎜

      생산 종료 시뮬레이션 메시지 손실됨🎜여기에는 두 가지 해결 방법이 있습니다. 🎜
      • 🎜메시지를 보낸 후 즉시 브로커를 닫습니다. 후자는 네트워크를 종료하지만 브로커가 닫힌 후에는 콘솔은 항상 오류를 보고합니다. 보내기 메시지도 500 오류를 보고했습니다. 🎜
      • 🎜존재하지 않는 스위치 보내기: 🎜
      • 🎜rrreee🎜결과: 🎜
        🎜[correlationData]:null
        [ack]false
        [cause] 채널 오류; 프로토콜 방법: #method(reply-code=404, reply-text=NOT_FOUND - 가상 호스트 '/', class-id=60, method-id=40에서 'myExchangexxxxx' 교환 없음)[전송 실패]🎜
        🎜전송 실패 시 메시지를 다시 시도할 수 있습니다🎜🎜스위치가 정확합니다. 존재하지 않는 대기열로 전송합니다.🎜🎜스위치가 메시지를 수신하고 성공 알림을 반환합니다. 및 콘솔 출력:🎜🎜【correlationData】:CorrelationData [id=7d468b47-b422-4523-b2a2-06b14aef073c]
        [ack]true
        [cause]null
        [ 성공적으로 전송됨]🎜 🎜스위치가 대기열을 찾지 못했고 실패 정보를 반환했습니다. 🎜
        🎜[메시지 전송 실패]
        [message]제품 메시지
        [replyCode]312🎜🎜RabbitMQ🎜🎜큐 지속성을 활성화하면 생성된 큐와 스위치가기본 구성이 지속됩니다. 먼저 큐와 스위치를 올바르게 설정하고 메시지가 큐에 저장되도록 소비 모니터링용 큐를 수정합니다. 🎜🎜큐의 지속성을 비지속성으로 수정: 🎜rrreee🎜메시지를 보낸 후 메시지는 큐에 저장된 다음 RabbitMQ를 다시 시작하고 메시지는 더 이상 존재하지 않습니다.
        대기열 지속성 설정: 🎜rrreee🎜다시 시작한 후에도 대기열의 메시지는 계속 존재합니다. 🎜🎜소비자 측🎜🎜소비자 측은 기본적으로 ack 자동 확인 모드를 시작합니다. 대기열 메시지가 소비자에게 수신되면 대기열의 메시지가 있는지 여부에 관계없이 자동으로 삭제됩니다. 소비자 측의 메시지. 따라서 소비자가 메시지를 성공적으로 소비할 수 있도록 하려면 자동 모드를 수동 확인 모드로 변경하세요. 🎜🎜application.yml 파일을 수정하세요🎜rrreee🎜메시지를 소비하고 받은 후 수동 확인이 필요합니다. 🎜rrreeerrreee🎜If 추가되지 않음: 🎜rrreee🎜두 개의 메시지 보내기 🎜🎜메시지가 수신된 후 확인이 없으며 다시 대기열에 넣습니다. 🎜🎜🎜🎜🎜프로젝트를 다시 시작하면 대기열 메시지가 다음으로 전송됩니다. 소비자에게 전달되지만 확인 확인이 없으면 계속해서 대기열에 다시 들어가게 됩니다. 🎜

        channel.basicAck를 추가한 후 프로젝트를 다시 시작하세요channel.basicAck 之后,再重启项目

        SpringBoot+RabbitMQ를 사용하여 안정적인 메시지 전송을 달성하는 방법

        队列消息就被删除了

        basicAck 方法最后一个参数 multiple 表示是删除之前的队列。

        multiple 设置成 true

        SpringBoot+RabbitMQ를 사용하여 안정적인 메시지 전송을 달성하는 방법

        SpringBoot+RabbitMQ를 사용하여 안정적인 메시지 전송을 달성하는 방법큐 메시지가 삭제됩니다

        🎜basicAck multiple 메서드의 마지막 매개변수는 삭제됨을 의미합니다. 삭제 전 큐. 🎜🎜multipletrue로 설정되고 모든 후속 대기열이 지워집니다 🎜🎜🎜🎜

    위 내용은 SpringBoot+RabbitMQ를 사용하여 안정적인 메시지 전송을 달성하는 방법의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

    본 웹사이트의 성명
    본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.

    핫 AI 도구

    Undresser.AI Undress

    Undresser.AI Undress

    사실적인 누드 사진을 만들기 위한 AI 기반 앱

    AI Clothes Remover

    AI Clothes Remover

    사진에서 옷을 제거하는 온라인 AI 도구입니다.

    Undress AI Tool

    Undress AI Tool

    무료로 이미지를 벗다

    Clothoff.io

    Clothoff.io

    AI 옷 제거제

    AI Hentai Generator

    AI Hentai Generator

    AI Hentai를 무료로 생성하십시오.

    인기 기사

    R.E.P.O. 에너지 결정과 그들이하는 일 (노란색 크리스탈)
    4 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
    R.E.P.O. 최고의 그래픽 설정
    4 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
    R.E.P.O. 아무도들을 수없는 경우 오디오를 수정하는 방법
    4 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
    R.E.P.O. 채팅 명령 및 사용 방법
    4 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌

    뜨거운 도구

    메모장++7.3.1

    메모장++7.3.1

    사용하기 쉬운 무료 코드 편집기

    SublimeText3 중국어 버전

    SublimeText3 중국어 버전

    중국어 버전, 사용하기 매우 쉽습니다.

    스튜디오 13.0.1 보내기

    스튜디오 13.0.1 보내기

    강력한 PHP 통합 개발 환경

    드림위버 CS6

    드림위버 CS6

    시각적 웹 개발 도구

    SublimeText3 Mac 버전

    SublimeText3 Mac 버전

    신 수준의 코드 편집 소프트웨어(SublimeText3)

    React와 RabbitMQ를 사용하여 안정적인 메시징 앱을 구축하는 방법 React와 RabbitMQ를 사용하여 안정적인 메시징 앱을 구축하는 방법 Sep 28, 2023 pm 08:24 PM

    React 및 RabbitMQ를 사용하여 안정적인 메시징 애플리케이션을 구축하는 방법 소개: 최신 애플리케이션은 실시간 업데이트 및 데이터 동기화와 같은 기능을 달성하기 위해 안정적인 메시징을 지원해야 합니다. React는 사용자 인터페이스 구축을 위한 인기 있는 JavaScript 라이브러리인 반면 RabbitMQ는 안정적인 메시징 미들웨어입니다. 이 기사에서는 React와 RabbitMQ를 결합하여 안정적인 메시징 애플리케이션을 구축하는 방법을 소개하고 구체적인 코드 예제를 제공합니다. RabbitMQ 개요:

    RabbitMQ를 사용하여 PHP에서 분산 메시지 처리를 구현하는 방법 RabbitMQ를 사용하여 PHP에서 분산 메시지 처리를 구현하는 방법 Jul 18, 2023 am 11:00 AM

    RabbitMQ를 사용하여 PHP에서 분산 메시지 처리를 구현하는 방법 소개: 대규모 애플리케이션 개발에서 분산 시스템은 일반적인 요구 사항이 되었습니다. 분산 메시지 처리는 작업을 여러 처리 노드에 분산하여 시스템의 효율성과 안정성을 향상시키는 패턴입니다. RabbitMQ는 AMQP 프로토콜을 사용하여 메시지 전달 및 처리를 구현하는 신뢰할 수 있는 오픈 소스 메시지 대기열 시스템입니다. 이 기사에서는 배포를 위해 PHP에서 RabbitMQ를 사용하는 방법을 다룹니다.

    SpringBoot와 SpringMVC의 비교 및 ​​차이점 분석 SpringBoot와 SpringMVC의 비교 및 ​​차이점 분석 Dec 29, 2023 am 11:02 AM

    SpringBoot와 SpringMVC는 모두 Java 개발에서 일반적으로 사용되는 프레임워크이지만 둘 사이에는 몇 가지 분명한 차이점이 있습니다. 이 기사에서는 이 두 프레임워크의 기능과 용도를 살펴보고 차이점을 비교할 것입니다. 먼저 SpringBoot에 대해 알아봅시다. SpringBoot는 Spring 프레임워크를 기반으로 하는 애플리케이션의 생성 및 배포를 단순화하기 위해 Pivotal 팀에서 개발되었습니다. 독립 실행형 실행 파일을 구축하는 빠르고 가벼운 방법을 제공합니다.

    Go에서 RabbitMQ 사용: 전체 가이드 Go에서 RabbitMQ 사용: 전체 가이드 Jun 19, 2023 am 08:10 AM

    최신 애플리케이션의 복잡성이 증가함에 따라 메시징은 강력한 도구가 되었습니다. 이 분야에서 RabbitMQ는 다양한 애플리케이션 간에 메시지를 전달하는 데 사용할 수 있는 매우 인기 있는 메시지 브로커가 되었습니다. 이 기사에서는 Go 언어에서 RabbitMQ를 사용하는 방법을 살펴보겠습니다. 이 가이드에서는 다음 내용을 다룹니다. RabbitMQ 소개 RabbitMQ 설치 RabbitMQ 기본 개념 Go에서 RabbitMQ 시작하기 RabbitMQ 및 Go

    SpringBoot+Dubbo+Nacos 개발 실습 튜토리얼 SpringBoot+Dubbo+Nacos 개발 실습 튜토리얼 Aug 15, 2023 pm 04:49 PM

    이 글에서는 dubbo+nacos+Spring Boot의 실제 개발에 대해 이야기하기 위해 자세한 예제를 작성하겠습니다. 이 기사에서는 이론적 지식을 너무 많이 다루지는 않지만 dubbo를 nacos와 통합하여 개발 환경을 신속하게 구축하는 방법을 설명하는 가장 간단한 예를 작성합니다.

    Golang과 RabbitMQ 간의 실시간 데이터 동기화 솔루션 Golang과 RabbitMQ 간의 실시간 데이터 동기화 솔루션 Sep 27, 2023 pm 10:41 PM

    Golang과 RabbitMQ 간의 실시간 데이터 동기화 솔루션 소개: 오늘날 인터넷의 대중화와 데이터 양의 폭발적인 증가로 인해 실시간 데이터 동기화가 점점 더 중요해지고 있습니다. 비동기 데이터 전송 및 데이터 동기화 문제를 해결하기 위해 많은 회사에서는 메시지 대기열을 사용하여 데이터의 실시간 동기화를 달성하기 시작했습니다. 이 글에서는 Golang과 RabbitMQ를 기반으로 한 실시간 데이터 동기화 솔루션을 소개하고 구체적인 코드 예시를 제공합니다. 1. RabbitMQ란 무엇인가요? 랍비

    go-zero와 RabbitMQ의 응용실습 go-zero와 RabbitMQ의 응용실습 Jun 23, 2023 pm 12:54 PM

    이제 점점 더 많은 회사들이 마이크로서비스 아키텍처 모델을 채택하기 시작하고 있으며 이 아키텍처에서 메시지 큐는 중요한 통신 방법이 되었으며 그 중 RabbitMQ가 널리 사용됩니다. Go 언어에서 go-zero는 최근 몇 년 동안 등장한 프레임워크로, 개발자가 메시지 대기열을 보다 쉽게 ​​사용할 수 있도록 다양한 실용적인 도구와 방법을 제공합니다. 아래에서는 실제 응용 프로그램을 기반으로 한 go-zero를 소개합니다. RabbitMQ의 응용실습. 1.RabbitMQ 개요Rabbit

    Golang RabbitMQ: 고가용성 메시지 대기열 시스템의 아키텍처 설계 및 구현 Golang RabbitMQ: 고가용성 메시지 대기열 시스템의 아키텍처 설계 및 구현 Sep 28, 2023 am 08:18 AM

    GolangRabbitMQ: 고가용성 메시지 대기열 시스템의 아키텍처 설계 및 구현에는 특정 코드 예제가 필요합니다. 소개: 인터넷 기술의 지속적인 발전과 광범위한 응용으로 인해 메시지 대기열은 현대 소프트웨어 시스템에서 없어서는 안될 부분이 되었습니다. 분리, 비동기 통신, 내결함성 처리 및 기타 기능을 구현하는 도구로서 메시지 큐는 분산 시스템에 대한 고가용성 및 확장성 지원을 제공합니다. 효율적이고 간결한 프로그래밍 언어인 Golang은 높은 동시성 및 고성능 시스템을 구축하는 데 널리 사용됩니다.

    See all articles