• RabbitMQ消息的可靠性


    一、RabbitMQ消息的可靠性

    1.1 RabbitMQ事务

    当在消息发送过程中添加了事务,处理效率降低几十倍甚至上百倍

    channel.txSelect();  //开启事务
    try{
        channel.basicPublish("ex4", "k1", null, msg.getBytes());
        System.out.println("发送:" + msg);
        channel.txCommit(); //提交事务
    }catch (Exception e){
        channel.txRollback(); //事务回滚
    }
    

    1.2 RabbitMQ消息确认和return机制

    image-20210112104753019

    消息确认机制:确认消息提供者是否成功发送消息到交换机

    return机制:确认消息是否成功的从交换机分发到队列

    1.2.1 普通Maven项目的消息确认
    • 普通confirm方式
    //1.发送消息之前开启消息确认
    channel.confirmSelect();
    
    channel.basicPublish("ex1", "a", null, msg.getBytes());
    
    //2.接收消息确认
    boolean b = channel.waitForConfirms(); 
    
    System.out.println("发送:" +(b?"成功":"失败"));
    
    • 批量confirm方式
    //1.发送消息之前开启消息确认
    channel.confirmSelect();
    
    //2.批量发送消息
    for (int i=0 ; i<10 ; i++){
        channel.basicPublish("ex1", "a", null, msg.getBytes());
    }
    
    //3.接收批量消息确认:发送的所有消息中,如果有一条是失败的,则所有消息发送直接失败,抛出IO异常
    boolean b = channel.waitForConfirms(); 
    
    • 异步confirm方式
    //发送消息之前开启消息确认
    channel.confirmSelect();
    
    //批量发送消息
    for (int i=0 ; i<10 ; i++){
        channel.basicPublish("ex1", "a", null, msg.getBytes());
    }
    
    //假如发送消息需要10s,waitForConfirms会进入阻塞状态
    //boolean b = channel.waitForConfirms();
    
    //使用监听器异步confirm
    channel.addConfirmListener(new ConfirmListener() {
        //参数1: long l  返回消息的表示
        //参数2: boolean b 是否为批量confirm
        public void handleAck(long l, boolean b) throws IOException {
            System.out.println("~~~~~消息成功发送到交换机");
        }
        public void handleNack(long l, boolean b) throws IOException {
            System.out.println("~~~~~消息发送到交换机失败");
        }
    });
    
    1.2.2 普通Maven项目的return机制
    • 添加return监听器
    • 发送消息是指定第三个参数为true
    • 由于监听器监听是异步处理,所以在消息发送之后不能关闭channel
    String msg = "Hello HuangDaoJun!";
    Connection connection = ConnectionUtil.getConnection();     //相当于JDBC操作的数据库连接
    Channel channel = connection.createChannel();               //相当于JDBC操作的statement
    
    //return机制:监控交换机是否将消息分发到队列
    channel.addReturnListener(new ReturnListener() {
        public void handleReturn(int i, String s, String s1, String s2,AMQP.BasicProperties basicProperties,byte[] bytes) throws IOException {
            //如果交换机分发消息到队列失败,则会执行此方法(用来处理交换机分发消息到队列失败的情况)
            System.out.println("*****"+i);  //标识
            System.out.println("*****"+s);  //
            System.out.println("*****"+s1); //交换机名
            System.out.println("*****"+s2); //交换机对应的队列的key
            System.out.println("*****"+new String(bytes));  //发送的消息
        }
    });
    
    //发送消息
    //channel.basicPublish("ex2", "c", null, msg.getBytes());
    channel.basicPublish("ex2", "c", true, null, msg.getBytes());
    

    1.3 在SpringBoot应用实现消息确认与return监听

    1.3.1 配置application.yml,开启消息确认和return监听
    spring:
      rabbitmq:
        publisher-confirm-type: simple  ## 开启消息确认模式
        publisher-returns: true		##使用return监听机制
    
    1.3.2 创建confirm和return监听
    @Component
    public class MsgConfirmAndReturn implements RabbitTemplate.ConfirmCallback,RabbitTemplate.ReturnCallback {
    
        Logger logger = LoggerFactory.getLogger(MsgConfirmAndReturn.class);
    
        @Resource
        private RabbitTemplate rabbitTemplate;
    
        @PostConstruct
        public void init(){
            rabbitTemplate.setConfirmCallback(this);
            rabbitTemplate.setReturnCallback(this);
        }
    
        @Override
        public void confirm(CorrelationData correlationData, boolean b, String s) {
            //此方法用于监听消息确认结果(消息是否发送到交换机)
            if(b){
                logger.info("-------消息成功发送到交换机");
            }else{
                logger.warn("-------消息发送到交换机失败");
            }
        }
    
        @Override
        public void returnedMessage(Message message, int i, String s, String s1, String s2) {
            //此方法用于return监听(当交换机分发消息到队列失败时执行)
            logger.warn("~~~~~~~交换机分发消息到队列失败");
        }
    }
    

    1.4 RabbitMQ的集群部署

    • 暂略
  • 相关阅读:
    linux网卡eth1如何修改为eth0
    rpm方式安装MySQL5.1.73
    quartz demo01
    chrome 搜索 jsonView
    判断是否十六进制格式字符串
    ip and port check 正则
    hadoop 遇到java.net.ConnectException: to 0.0.0.0:10020 failed on connection
    hadoop2.4.1 伪分布
    R 包
    使用pt-heartbeat检测主从复制延迟
  • 原文地址:https://www.cnblogs.com/coderD/p/14265849.html
Copyright © 2020-2023  润新知