当前位置:首页 > 科技  > 软件

消息队列批量收发消息,请避开这五个坑!

来源: 责编: 时间:2023-11-30 09:26:20 307观看
导读大家好,我是君哥。使用消息队列时,为了提高生产和消费的性能,有时会开启批量处理。在生产端,生产者发送的消息先发送到一个消息列表,积累到一定的消息量之后再批量发送给 Broker,如下图:在消费端,消费者拉取消息后先不立即处

3dK28资讯网——每日最新资讯28at.com

大家好,我是君哥。3dK28资讯网——每日最新资讯28at.com

使用消息队列时,为了提高生产和消费的性能,有时会开启批量处理。3dK28资讯网——每日最新资讯28at.com

在生产端,生产者发送的消息先发送到一个消息列表,积累到一定的消息量之后再批量发送给 Broker,如下图:3dK28资讯网——每日最新资讯28at.com

3dK28资讯网——每日最新资讯28at.com

在消费端,消费者拉取消息后先不立即处理,而是把消息转存到一个内存队列或数据库,由业务线程去处理,如下图:3dK28资讯网——每日最新资讯28at.com

3dK28资讯网——每日最新资讯28at.com

无论是生产者做批量发送,还是消费者做批量处理,都需要考虑使用批量消息的业务场景,避免踩坑。下面看一下批量操作可能会遇到哪些坑。3dK28资讯网——每日最新资讯28at.com

批量大小

当生产者采用批量发送的方式来提高发送性能时,一定要考虑发送消息的批量大小。下面是 RocketMQ 批量发送的官方示例:3dK28资讯网——每日最新资讯28at.com

String topic = "BatchTest";List<Message> messages = new ArrayList<>();messages.add(new Message(topic, "TagA", "OrderID001", "Hello world 0".getBytes()));messages.add(new Message(topic, "TagA", "OrderID002", "Hello world 1".getBytes()));messages.add(new Message(topic, "TagA", "OrderID003", "Hello world 2".getBytes()));try {    producer.send(messages);} catch (Exception e) {    e.printStackTrace();    //handle the error}

RocketMQ 默认消息大小是 4M,由 maxMessageSize 参数控制,如果批量消息大小超过 maxMessageSize,则会抛出异常。3dK28资讯网——每日最新资讯28at.com

如果遇到消息大小超过 maxMessageSize 的情况时,可以用下面方法进行处理:3dK28资讯网——每日最新资讯28at.com

  • 把这个参数改大,但需要考虑 Broker 的性能和网络带宽;
  • 将消息进行拆分后分批发送;
  • 对消息进行压缩处理。

RabbitMQ 相关的 API 则提供了更加灵活的批量控制,对消息数量和消息大小都做了控制,下面看一下源码:3dK28资讯网——每日最新资讯28at.com

3dK28资讯网——每日最新资讯28at.com

3dK28资讯网——每日最新资讯28at.com

幂等

消费端可以批量拉取消息进行消费,这样可以减少拉取消息时的 RPC 次数,提升消费性能。比如在 RocketMQ 中,可以通过 Consumer 中的 pullBatchSize 来设置一次拉取的消息数量,通过 consumeMessageBatchMaxSize 参数来设置一次消费的消息数量。3dK28资讯网——每日最新资讯28at.com

但需要注意的是,如果批量消息中一条消息消费失败了,这一批消息都需要进行重试,已经消费成功的消息会被重复消费,带来业务问题。3dK28资讯网——每日最新资讯28at.com

为了不对业务造成影响,必须考虑幂等。一个简单的方法是在消息中增加全局唯一 id 属性,对消息消费结果进行记录,消费成功后保存 id。这样在消费消息之前先查询是否存在消费成功的记录,如果存在则直接返回处理成功。3dK28资讯网——每日最新资讯28at.com

时延

在使用消息队列进行批量操作时,必须要考虑到时延问题。比如我们设置一个批次 100 条消息,积累够 100 条消息后再发送,在消息量小的情况下,可能积累够 100 条消息会很长时间,导致消费端拉取到一条消息时延很大。3dK28资讯网——每日最新资讯28at.com

虽然消息队列的一个重要作用是削峰填谷,但在一些场景下,对消息的实时性也有要求。比如在车联网的充电场景,车联网平台需要实时感知充电桩的状态,如果充电桩积累够一批消息再上报平台,平台获取到的状态会不准确,如果心跳消息延时太久,平台会认为充电桩离线。3dK28资讯网——每日最新资讯28at.com

对于有时延要求又需要批量操作的场景,可以设置一个超时时间,超时后即使消息数量不够,也会发送出去。看下 RabbitMQ 的处理:3dK28资讯网——每日最新资讯28at.com

public synchronized void send(String exchange, String routingKey, Message message, CorrelationData correlationData)  throws AmqpException { if (correlationData != null) {  //...  super.send(exchange, routingKey, message, correlationData); } else {  if (this.scheduledTask != null) {   this.scheduledTask.cancel(false);  }  MessageBatch batch = this.batchingStrategy.addToBatch(exchange, routingKey, message);  if (batch != null) {   super.send(batch.getExchange(), batch.getRoutingKey(), batch.getMessage(), null);  }  //这里获取到超时时间,到达超时时间后使用定时器将消息发送出去  Date next = this.batchingStrategy.nextRelease();  if (next != null) {   this.scheduledTask = this.scheduler.schedule((Runnable) () -> releaseBatches(), next);  } }}

可靠性

使用批处理一定要考虑可靠性的问题。3dK28资讯网——每日最新资讯28at.com

在消费端,消费者批量拉取一批消息后把消息暂存到一个内存临时队列,然后多线程去临时队列消费消息,如果服务宕机,临时队列中的消息会丢失。3dK28资讯网——每日最新资讯28at.com

为了避免宕机引发的损失,可以拉取一批消息后保存到数据库,然后给 Broker 返回 ACK,之后业务代码去数据库查询消息并消费,不过要考虑数据库大事务、锁竞争等问题。3dK28资讯网——每日最新资讯28at.com

当然,对于一些消息丢失不敏感的场景,比如日志收集之类的,可靠性这个指标是不用太关注的。3dK28资讯网——每日最新资讯28at.com

特殊场景

因为批量消息有一些复杂性,消息队列的部分特性不支持。3dK28资讯网——每日最新资讯28at.com

事务消息

批量消息会增加消息重试的难度,所以对于事务消息,建议使用单条消息,一条消息对应一个事务。3dK28资讯网——每日最新资讯28at.com

顺序消息

顺序消息的实现思路一般是生产者将消息发送到同一个分区,消费者绑定这个分区并使用单线程消费这个分区的消息。如果对同一个 Topic 下的同一个分区来实现批量发送,难度会增大。所以建议顺序消息使用单条消息进行发送。3dK28资讯网——每日最新资讯28at.com

延时消息

如果延时消息使用批量进行发送,这一批消息的延时时间必须相同,同时要考虑批量消息的超时时间,超时时间太大会影响延时时间的准确性,生产端实现复杂度大大增加。3dK28资讯网——每日最新资讯28at.com

总结

使用批量消息,在一定程度上可以提高性能和吞吐量,但是确实也会存在一些问题,使用的时候要结合业务场景避开这些坑。3dK28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-35258-0.html消息队列批量收发消息,请避开这五个坑!

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: 利用属性选择器对外部链接进行样式设计

下一篇: 全网最详细的OpenFeign讲解,肯定有你不知道的

标签:
  • 热门焦点
  • 掘力计划第 20 期:Flutter 混合开发的混乱之治

    在掘力计划系列活动第20场,《Flutter 开发实战详解》作者,掘金优秀作者,Github GSY 系列目负责人恋猫的小郭分享了Flutter 混合开发的混乱之治。Flutter 基于自研的 Skia 引擎
  • 如何通过Python线程池实现异步编程?

    线程池的概念和基本原理线程池是一种并发处理机制,它可以在程序启动时创建一组线程,并将它们置于等待任务的状态。当任务到达时,线程池中的某个线程会被唤醒并执行任务,执行完任
  • 多线程开发带来的问题与解决方法

    使用多线程主要会带来以下几个问题:(一)线程安全问题  线程安全问题指的是在某一线程从开始访问到结束访问某一数据期间,该数据被其他的线程所修改,那么对于当前线程而言,该线程
  • 微信语音大揭秘:为什么禁止转发?

    大家好,我是你们的小米。今天,我要和大家聊一个有趣的话题:为什么微信语音不可以转发?这是一个我们经常在日常使用中遇到的问题,也是一个让很多人好奇的问题。让我们一起来揭开这
  • 只需五步,使用start.spring.io快速入门Spring编程

    步骤1打开https://start.spring.io/,按照屏幕截图中的内容创建项目,添加 Spring Web 依赖项,并单击“生成”按钮下载 .zip 文件,为下一步做准备。请在进入步骤2之前进行解压。图
  • 19个 JavaScript 单行代码技巧,让你看起来像个专业人士

    今天这篇文章跟大家分享18个JS单行代码,你只需花几分钟时间,即可帮助您了解一些您可能不知道的 JS 知识,如果您已经知道了,就当作复习一下,古人云,温故而知新嘛。现在,我们就开始今
  • 每天一道面试题-CPU伪共享

    前言:了不起:又到了每天一到面试题的时候了!学弟,最近学习的怎么样啊 了不起学弟:最近学习的还不错,每天都在学习,每天都在进步! 了不起:那你最近学习的什么呢? 了不起学弟:最近在学习C
  • 10天营收超1亿美元,《星铁》比《原神》差在哪?

    来源:伯虎财经作者:陈平安即便你没玩过《原神》,你一定听说过的它的大名。恨它的人把《原神》开服那天称作是中国游戏史上最黑暗的一天,有粉丝因为索尼在PS平台上线《原神》,怒而
  • 华为Mate 60系列用上可变灵动岛:正式版体验将会更出色

    这段时间以来,关于华为新旗舰的爆料日渐密集。据此前多方爆料,今年华为将开始恢复一年双旗舰战略,除上半年推出的P60系列外,往年下半年的Mate系列也将
Top