rabbitMq高级知识

  • 消息的可靠性
    • 消息的可靠投递【生产者】
      • 消息在发送到RabbitMQ时,一定能够发送成功吗?是否会丢失?怎么确保丢失不丢失?
      • 实现方案:confirm 确认模式
      • rabbitmq 整个消息投递的路径
        • producer--->rabbitmq broker--->exchange--->queue--->consumer
        • 将利用这两个 callback 控制消息的可靠性投递
          • 消息从 producer 到 exchange 则会返回一个 confirmCallback
          • 消息从 exchange-->queue 投递失败则会返回一个 returnCallback
      • 具体实现
        • confirm 确认模式
          • 开启消息确认模式
          • spring-rabbitmq-producer.xml
            • 定义队列
              • <rabbit:queue id="test_queue_confirm" name="test_queue_confirm"></rabbit:queue>
            • 定义交换机
              • <rabbit:direct-exchange name="test_exchange_confirm">
                    <rabbit:bindings>
                        <rabbit:binding queue="test_queue_confirm" key="confirm"></rabbit:binding>
                    </rabbit:bindings>
                </rabbit:direct-exchange>
            • 开启确认模式
              •  <rabbit:connection-factory id="connectionFactory" host="${rabbitmq.host}"
                                               port="${rabbitmq.port}"
                                               username="${rabbitmq.username}"
                                               password="${rabbitmq.password}"
                                               virtual-host="${rabbitmq.virtual-host}"
                                               publisher-confirms="true"/>
            • 设置回调函数
              • 使用rabbitTemplate.setConfirmCallback设置回调函数。当消息发送到exchange后回
                调confirm方法。在方法中判断ack,如果为true,则发送成功,如果为false,则发
                送失败,需要处理

                /**
                 * 确认模式:
                 * 步骤:
                 * 1. 确认模式开启:ConnectionFactory中开启publisher-confirms="true"
                 * 2. 在rabbitTemplate定义ConfirmCallBack回调函数
                 */
                @Test
                public void testConfirm() {
                
                    //2. 定义回调
                    rabbitTemplate.setConfirmCallback(new RabbitTemplate.ConfirmCallback() {
                        /**
                         *
                         * @param correlationData 相关配置信息
                         * @param ack   exchange交换机 是否成功收到了消息。true 成功,false代表失败
                         * @param cause 失败原因
                         */
                        @Override
                        public void confirm(CorrelationData correlationData, boolean ack, String cause) {
                            System.out.println("confirm方法被执行了....");
                
                            if (ack) {
                                //接收成功
                                System.out.println("接收成功消息" + cause);
                            } else {
                                //接收失败
                                System.out.println("接收失败消息" + cause);
                                //做一些处理,让消息再次发送。
                            }
                        }
                    });
                
                    //3. 发送消息
                    rabbitTemplate.convertAndSend("test_exchange_confirm", "confirm", "message confirm....");
                }
        • return 退回模式
          • 开启消息退回模式
          • spring-rabbitmq-producer.xml
            • 定义队列
              • <rabbit:queue id="test_queue_confirm" name="test_queue_confirm"></rabbit:queue>
            • 定义交换机
              • <rabbit:direct-exchange name="test_exchange_confirm">
                    <rabbit:bindings>
                        <rabbit:binding queue="test_queue_confirm" key="confirm"></rabbit:binding>
                    </rabbit:bindings>
                </rabbit:direct-exchange>
            • 开启确认模式
              • <!-- 定义rabbitmq connectionFactory -->
                    <rabbit:connection-factory id="connectionFactory" host="${rabbitmq.host}"
                                               port="${rabbitmq.port}"
                                               username="${rabbitmq.username}"
                                               password="${rabbitmq.password}"
                                               virtual-host="${rabbitmq.virtual-host}"
                                               publisher-confirms="true"
                    publisher-returns="true"/>
            • 设置退回函数
              • /**
                 * 回退模式: 当消息发送给Exchange后,Exchange路由到Queue失败是 才会执行 ReturnCallBack
                 * 步骤:
                 * 1. 开启回退模式:publisher-returns="true"
                 * 2. 设置ReturnCallBack
                 * 3. 设置Exchange处理消息的模式:
                 * 1. 如果消息没有路由到Queue,则丢弃消息(默认)
                 * 2. 如果消息没有路由到Queue,返回给消息发送方ReturnCallBack
                 */
                @Test
                public void testReturn() {
                    //设置交换机处理失败消息的模式,才能回调 returnedMessage 方法
                    rabbitTemplate.setMandatory(true);
                
                    //2.设置ReturnCallBack
                    rabbitTemplate.setReturnCallback(new RabbitTemplate.ReturnCallback() {
                        /**
                         *
                         * @param message   消息对象
                         * @param replyCode 错误码
                         * @param replyText 错误信息
                         * @param exchange  交换机
                         * @param routingKey 路由键
                         */
                        @Override
                        public void returnedMessage(Message message, int replyCode, String replyText, String exchange, String routingKey) {
                            System.out.println("return 执行了....");
                            System.out.println(message);
                            System.out.println(replyCode);
                            System.out.println(replyText);
                            System.out.println(exchange);
                            System.out.println(routingKey);
                            //处理
                        }
                    });
                    //3. 发送消息
                    rabbitTemplate.convertAndSend("test_exchange_confirm", "confirm", "message confirm....");
                }
              • 只能direct和topic模式使用,returns退出模式和routinskey有关,confirm模式和exchage有关
    • Consumer Ack【消费者】
      • 确认方式
        • 其中自动确认是指,当消息一旦被Consumer接收到,则自动确认收到,并将相应 message 从 RabbitMQ 的
          消息缓存中移除。但是在实际业务处理中,很可能消息接收到,业务处理出现异常,那么该消息就会丢失。如
          果设置了手动确认方式,则需要在业务处理成功后,调用channel.basicAck(),手动签收,如果出现异常,则
          调用channel.basicNack()方法,让其自动重新发送消息。

        • 自动确认:acknowledge="none"
        • 手动确认:acknowledge="manual"
      • 具体实现
        • spring-rabbitmq-consumer.xml
          • 配置监听器容器,开且手动签收
            • <!--定义监听器容器-->
              <rabbit:listener-container connection-factory="connectionFactory" acknowledge="manual">
                  <rabbit:listener ref="ackListener" queue-names="test_queue_confirm"></rabbit:listener>
              </rabbit:listener-container>
          • 开启包扫描
            • <context:component-scan base-package="com.itheima.listener" />
          • 编写监听器
            • @Component
              public class AckListener implements ChannelAwareMessageListener {
              
                  @Override
                  public void onMessage(Message message, Channel channel) throws Exception {
              	   Thread.sleep(1000);
              	    //消息的Tag
                      long deliveryTag = message.getMessageProperties().getDeliveryTag();
              
                      try {
                          //1.接收转换消息
                          System.out.println(new String(message.getBody()));
              
                          //2. 处理业务逻辑
                          System.out.println("处理业务逻辑...");
                          int i = 3/0;//出现错误
                          //3. 手动签收
                          channel.basicAck(deliveryTag,true);
                      } catch (Exception e) {
                          //e.printStackTrace();
              
                          //4.拒绝签收
                          /*
                          第三个参数:requeue:重回队列。如果设置为true,则消息重新回到queue,broker会重新发送该消息给消费端
                           */
                          channel.basicNack(deliveryTag,true,true);
                          //channel.basicReject(deliveryTag,true);
                      }
                  }
              }
              
    • 死信队列
      • 消费端限流
        • 2020-03-19_213209
        • 具体实现
          • 前提:必须手动签收
          • 修改消费者配置:spring-rabbitmq-consumer.xml
            • <!--定义监听器容器
                  acknowledge 签收方式
                      none 自动签收
                      auto 根据异常决定
                      manual 手动签收
                  prefetch="1"
                      每次从rabbitmq中获取的消息数量
              -->
              <rabbit:listener-container connection-factory="connectionFactory" acknowledge="manual" prefetch="1">
                  <rabbit:listener ref="qosListener" queue-names="test_queue_confirm"></rabbit:listener>
              </rabbit:listener-container>
              
          • 创建监听器类
            • /**
               * Consumer 限流机制
               *  1. 确保ack机制为手动确认。
               *  2. listener-container配置属性
               *      perfetch = 1,表示消费端每次从mq拉去一条消息来消费,直到手动确认消费完毕后,才会继续拉去下一条消息。
               */
              
              @Component
              public class QosListener implements ChannelAwareMessageListener {
              
                  @Override
                  public void onMessage(Message message, Channel channel) throws Exception {
              
                      Thread.sleep(1000);
                      //1.获取消息
                      System.out.println(new String(message.getBody()));
              
                      //2. 处理业务逻辑
              
                      //3. 签收
                      channel.basicAck(message.getMessageProperties().getDeliveryTag(),true);
              
                  }
              }
          • 生产者创建测试方法,发送多条消息
            • @Test
              public void testSend() {
              
                  for (int i = 0; i < 10; i++) {
                      // 发送消息
                      rabbitTemplate.convertAndSend("test_exchange_confirm", "confirm", "message confirm....");
                  }
              }
    • ttl
      • 全称 Time To Live(存活时间/过期时间)
      • 当消息到达存活时间后,还没有被消费,会被自动清除
      • RabbitMQ可以对消息设置过期时间,也可以对整个队列(Queue)设置过期时间。
      • 管理控制台实现
        • 2020-03-19_214211
      • 代码实现
        • 创建队列与交换机
          • spring-rabbitmq-producer.xml
            • <!--ttl-->
              <rabbit:queue name="test_queue_ttl" id="test_queue_ttl">
                  <!--设置queue的参数-->
                  <rabbit:queue-arguments>
                      <!--x-message-ttl指队列的过期时间-->
                      <entry key="x-message-ttl" value="100000" value-type="java.lang.Integer"></entry>
                  </rabbit:queue-arguments>
              
              </rabbit:queue>
              
              <rabbit:topic-exchange name="test_exchange_ttl" >
                  <rabbit:bindings>
                      <rabbit:binding pattern="ttl.#" queue="test_queue_ttl"></rabbit:binding>
                  </rabbit:bindings>
              </rabbit:topic-exchange>
              
          • 消息的过期设置
            • /**
               * TTL:过期时间
               *  1. 队列统一过期
               *
               *  2. 消息单独过期
               *
               * 如果设置了消息的过期时间,也设置了队列的过期时间,它以时间短的为准。
               * 队列过期后,会将队列所有消息全部移除。
               * 消息过期后,只有消息在队列顶端,才会判断其是否过期(移除掉)
               *
               */
              @Test
              public void testTtl2() {
              
                // 消息后处理对象,设置一些消息的参数信息
                  MessagePostProcessor messagePostProcessor = new MessagePostProcessor() {
              
                      @Override
                      public Message postProcessMessage(Message message) throws AmqpException {
                          //1.设置message的信息
                          message.getMessageProperties().setExpiration("5000");//消息的过期时间
                          //2.返回该消息
                          return message;
                      }
                  };
              
                  //消息单独过期
                  //rabbitTemplate.convertAndSend("test_exchange_ttl", "ttl.hehe", "message ttl....",messagePostProcessor);
              
                  for (int i = 0; i < 10; i++) {
                      if(i == 5){
                          //消息单独过期
                          //消息过期后,只有消息在队列顶端,才会判断其是否过期(移除掉),而这里5并没有在顶端,所以不会去判断过期时间
                          rabbitTemplate.convertAndSend("test_exchange_ttl", "ttl.hehe", "message ttl....",messagePostProcessor);
                      }else{
                          //不过期的消息
                          rabbitTemplate.convertAndSend("test_exchange_ttl", "ttl.hehe", "message ttl....");
              
                      }
              
                  }
              
              }
      • 设置队列过期时间使用参数:x-message-ttl,单位:ms(毫秒),会对整个队列消息统一过期
      • 设置消息过期时间使用参数:expiration。单位:ms(毫秒),当该消息在队列头部时(消费时),会单独判断这一消息是否过期。
      • 如果两者都进行了设置,以时间短的为准。
    • 死信队列
      • 消息成为死信的三种情况
        • 队列消息长度到达限制
        • 消费者拒接消费消息,basicNack/basicReject,并且不把消息重新放入原目标队列,requeue=false;
        • 原队列存在消息过期设置,消息到达超时时间未被消费;
      • 代码实现
        • 创建死信队列与交换机
          • spring-rabbitmq-producer.xml
            • <!--
                 2. 声明死信队列(queue_dlx)和死信交换机(exchange_dlx)
              -->
              <rabbit:queue name="queue_dlx" id="queue_dlx"></rabbit:queue>
              <rabbit:topic-exchange name="exchange_dlx">
                  <rabbit:bindings>
                      <rabbit:binding pattern="dlx.#" queue="queue_dlx"></rabbit:binding>
                  </rabbit:bindings>
              </rabbit:topic-exchange>
        • 正常队列绑定死信交换机
          • spring-rabbitmq-producer.xml
            • <!--
                  1. 声明正常的队列(test_queue_dlx)和交换机(test_exchange_dlx)
              -->
              <rabbit:queue name="test_queue_dlx1" id="test_queue_dlx1">
                  <!--3. 正常队列绑定死信交换机-->
                  <rabbit:queue-arguments>
                      <!--3.1 x-dead-letter-exchange:死信交换机名称-->
                      <entry key="x-dead-letter-exchange" value="exchange_dlx" />
                      <!--3.2 x-dead-letter-routing-key:发送给死信交换机的routingkey-->
                      <entry key="x-dead-letter-routing-key" value="dlx.hehe" />
                      <!--4.1 设置队列的过期时间 ttl-->
                      <entry key="x-message-ttl" value="10000" value-type="java.lang.Integer" />
                      <!--4.2 设置队列的长度限制 max-length -->
                      <entry key="x-max-length" value="10" value-type="java.lang.Integer" />
                  </rabbit:queue-arguments>
              </rabbit:queue>
              <rabbit:topic-exchange name="test_exchange_dlx">
                  <rabbit:bindings>
                      <rabbit:binding pattern="test.dlx.#" queue="test_queue_dlx"></rabbit:binding>
                  </rabbit:bindings>
              </rabbit:topic-exchange>
              
    • 延迟队列
      • 即消息进入队列后不会立即被消费,只有到达指定时间后,才会被消费
      • TTL+死信队列 组合实现延迟队列的效果
      • 实现方式
        • 定时器
        • 延迟队列
      • 实现流程图
        • 2020-03-20_102904
      • RabbitMQ没有提供延迟队列功能,但是可以使用 : TTL + DLX 来实现延迟队列效果

posted on 2020-03-19 15:38  赟麟  阅读(183)  评论(0编辑  收藏  举报

导航