一、MessageQueueSelector 详解

MessageQueueSelector 是 RocketMQ 提供的一个接口,用于自定义消息发送时的队列选择策略。 通过实现该接口,

开发者可以控制消息被发送到 Topic 的哪个队列(MessageQueue),从而支持 顺序消息、负载均衡、业务隔离 等高级场景

1、接口定义

2、使用场景

A、顺序消息(关键场景)

  • 需求:同一业务键(如订单 ID)的消息必须发送到同一队列,保证消费顺序

  • 实现:

     // 示例:按订单ID哈希选择队列,确保同一订单的消息进入同一队列
    SendResult result = producer.send(msg, new MessageQueueSelector() {
    @Override
    public MessageQueue select(List<MessageQueue> queues, Message msg, Object arg) {
    String orderId = (String) arg; // arg 传入订单ID
    int index = Math.abs(orderId.hashCode()) % queues.size();
    return queues.get(index);
    }
    }, "订单123"); // 将订单ID作为 arg 传入

B、负载均衡优化

  • 需求:避免热点队列,根据业务特征分散消息

  • 实现:

     // 示例:按用户ID轮询选择队列
    SendResult result = producer.send(msg, (mqs, msg, arg) -> {
    long userId = (Long) arg;
    return mqs.get((int) (userId % mqs.size()));
    }, 12345L);

C、业务隔离

  • 需求:不同业务类型的消息路由到不同队列

  • 实现:

     // 示例:按消息标签(Tag)选择队列
    SendResult result = producer.send(msg, (mqs, msg, arg) -> {
    String tag = msg.getTags();
    if ("PAYMENT".equals(tag)) {
    return mqs.get(0); // 支付消息固定发到队列0
    } else {
    return mqs.get(1); // 其他消息发到队列1
    }
    }, null);

二、核心注意事项

1、队列数量稳定性

  • 问题:如果动态增加 Topic 的队列数,可能导致哈希结果不一致,破坏顺序消息

  • 解决:

    • 提前规划队列数(如固定 16 个队列),避免扩容

    • 如需扩容,需停机迁移或使用一致性哈希算法

2、参数传递

  • arg 的作用:通过 send() 方法的第三个参数传递业务标识(如订单ID),在 select() 中使用

3、异常处理

  • 空队列问题:如果 queues 为空(如 Topic 未创建),需抛出异常或降级处理

4、内置实现类

RocketMQ 提供了几个常用的内置选择器:

使用示例:

5、完整代码示例

6、最佳实践

  • 顺序消息:使用业务键(如订单ID)作为 arg,确保相同键的消息进入同一队列。

  • 性能优化:避免在 select() 中执行复杂计算(如数据库查询)。

  • 监控:记录队列选择分布,避免数据倾斜(如某些队列压力过大)。

详细介绍MessageQueueSelector的更多相关文章

  1. [No0000A7]批处理经常用到的变量及批处理>NUL详细介绍

    绝对路径是指调用绝对的程序位置的路径,例如: start C:\Windows\test.exe 相对路径是文件改变路径以后还会按照变量的路径所在位置去调用,例如: start %WINDIR%\te ...

  2. linux配置网卡IP地址命令详细介绍及一些常用网络配置命令

    linux配置网卡IP地址命令详细介绍及一些常用网络配置命令2010-- 个评论 收藏 我要投稿 Linux命令行下配置IP地址不像图形界面下那么方 便,完全需要我们手动配置,下面就给大家介绍几种配置 ...

  3. _MSC_VER详细介绍

    _MSC_VER详细介绍 转自:http://www.cnblogs.com/braver/articles/2064817.html _MSC_VER是微软的预编译控制. _MSC_VER可以分解为 ...

  4. php CGI、Fastcgi、PHP-FPM的详细介绍与之间的关系

    以下PHP CGI.Fastcgi.PHP-FPM的一些信息归纳和汇总----->详细介绍与之间的关系 一:CGI是干嘛的?CGI是为了保证web server传递过来的数据是标准格式的 web ...

  5. RabbitMQ消息队列(一): Detailed Introduction 详细介绍

     http://blog.csdn.net/anzhsoft/article/details/19563091 RabbitMQ消息队列(一): Detailed Introduction 详细介绍 ...

  6. doT.js详细介绍

    doT.js详细介绍   doT.js特点是快,小,无依赖其他插件. 官网:http://olado.github.iodoT.js详细使用介绍 使用方法:{{= }} for interpolati ...

  7. Linux截屏工具scrot用法详细介绍

    Scrot是Linux命令行中使用的截图工具,能够进行全屏.选取等操作,下面小编将针对Scrot截图工具的用法给大家做个详细介绍,通过操作实例来学习Scrot的使用.   在Linux中安装Scrot ...

  8. Oracle Merge into 详细介绍

    Oracle Merge into 详细介绍 /*Merge into 详细介绍MERGE语句是Oracle9i新增的语法,用来合并UPDATE和INSERT语句.通过MERGE语句,根据一张表或子查 ...

  9. cPage分页详细介绍

    asp.net中各种数据控件,datalist.gridview.Repeater等分页是最常用的功能,几乎任何一个B/S项目,无论是系统还是网站都会用到.分页时,读取整个数据,直接绑定到控件,都可以 ...

  10. 【转载】硬盘MBR详细介绍

    原文地址:http://blog.chinaunix.net/uid-15007890-id-106892.html 硬盘MBR详细介绍      硬盘是现在计算机上最常用的存储器之一.我们都知道,计 ...

随机推荐

  1. biancheng-Spring

    目录http://c.biancheng.net/spring/spring-abc.html 1Spring是什么2Spring体系结构3Spring开发环境搭建4第一个Spring程序5Sprin ...

  2. 将github个人访问令牌与TortoiseSVN一起使用

    最近用TortoiseSVN提交到Github身份验证,总是提示无效的用户名密码,反复确认密码没输入错的.但是就是提交不了(能获取). 报错如下: 错误: No more credentials or ...

  3. linux:搭建Drupal

    了解 Drupal 是使用 PHP 语言编写的开源内容管理框架(CMF),由内容管理系统(CMS)及 PHP 开发框架(Framework)共同构成.Drupal 具备强大的定制化开发能力,您可使用 ...

  4. Linux通配符和转移字符(扩展匹配文件名)、man帮助文档的使用

  5. ubuntu配置笔记

    一.ubuntu的mnt/hgfs下共享文件夹设置 1.确认VMware Tools和共享目录设定已经完成,安装vmware tool sudo apt-get install open-vm-too ...

  6. DeepSeek,你是懂.NET的!

    这两天火爆出圈的话题,除了过年,那一定是DeepSeek!你是否也被刷屏了? DeepSeek 是什么 DeepSeek是一款由国内人工智能公司研发的大型语言模型,拥有强大的自然语言处理能力,能够理解 ...

  7. Markdown编辑环境搭建

    摘要:这篇文章主要介绍我本人搭建Markdown编辑环境的全过程,并记录下在这个过程中所遇到的问题及解决方法. 一.VS Code安装 进入VS Code官网:VS Code官网 然后根据自己的电脑下 ...

  8. UpdateHub-一款好看且免费开源的Windows软件更新检测工具

    UpdateHub 是一款简化计算机上软件更新的应用程序.用户友好的界面允许您快速检查和安装操作系统和应用程序的可用更新. 通过这个应用,你可以快速地查看设备上安装的所有软件的更新,包括操作系统和应用 ...

  9. ABAQUS阻尼设置

    结构阻尼 瑞利阻尼

  10. 对接服务升级后仅支持tls1.2,jdk1.7默认使用tls1.0,导致调用失败

    背景 如标题所说,我手里维护了一个重要的老项目,使用jdk1.7,里面对接了很多个第三方服务,协议多种多样,其中涉及http/https的,调用方式也是五花八门,比如:commons-httpclie ...