基于Redis实现消息队列

基于Redis实现消息队列

1.业务场景

假设在没有专业消息中间件的情况下,又要通过消息队列去解耦。redis是个更好的选择。

2.实现方式

简要说明实现方式,这里只做个大概的概括

  • 发布与订阅(缺点:典型的一对一,不支持多个消费者公平消费消息,消息无法持久化,如果出现网络断开、Redis 宕机等,消息就会被丢弃等问题)

  • list队列(缺点:没有很好 ACK 机制,没有 ConsumerGroup 消费组,不支持一对多消费等问题)

  • stream队列(推荐)官方:https://redis.io/docs/data-types/streams/

3.概念

Redis5.0带来了Stream类型。其实就是Redis对消息队列(MQ,Message Queue)的完善实现。

主要有几个概念:

1.消费者组(Consumer Group):一个消费组有多个消费者(Consumer), 这些消费者之间是竞争关系。也就是说不会出现重复消费的场景。

2.pending_ids :消费者(Consumer)的状态变量,作用是维护消费者的未确认的 id。 pending_ids 记录了当前已经被客户端读取的消息,但是还没有 ack (Acknowledge character:确认字符)。

3.last_delivered_id :游标,每个消费组会有个游标 last_delivered_id,任意一个消费者读取了消息都会使游标 last_delivered_id 往前移动。

4.消息ID: 消息ID的形式是timestampInMillis-sequence,例如1527846880572-5

这里简要贴出Redis中Stream操作的相关指令

其实像代码,都是基于命令的高度封装

消息队列相关命令:

  • XADD - 添加消息到末尾

  • XTRIM - 对流进行修剪,限制长度

  • XDEL - 删除消息

  • XLEN - 获取流包含的元素数量,即消息长度

  • XRANGE - 获取消息列表,会自动过滤已经删除的消息

  • XREVRANGE - 反向获取消息列表,ID 从大到小

  • XREAD - 以阻塞或非阻塞方式获取消息列表

消费者组相关命令:

  • XGROUP CREATE - 创建消费者组

  • XREADGROUP GROUP - 读取消费者组中的消息

  • XACK - 将消息标记为"已处理"

  • XGROUP SETID - 为消费者组设置新的最后递送消息ID

  • XGROUP DELCONSUMER - 删除消费者

  • XGROUP DESTROY - 删除消费者组

  • XPENDING - 显示待处理消息的相关信息

  • XCLAIM - 转移消息的归属权

  • XINFO - 查看流和消费者组的相关信息;

  • XINFO GROUPS - 打印消费者组的信息;

  • XINFO STREAM - 打印流信息

4.代码实现

stream相关配置,这里主要配置消费组和消费者相关信息,以及消息的监听机制

@Slf4j
@Configuration
public class RedisStreamConfig {
    @Autowired
    private MyListener myListener;

    @Autowired
    private StringRedisTemplate redisTemplate;

    /**
     * 实际生产环境中  我们应该把消费者组等信息  写入配置环境中 
     */
//    @Autowired
//    private StreamProperty streamProperty;

    /**
     * 收到消息后不自动确认,需要用户选择合适的时机确认
     * 当某个消息被ACK,PEL列表就会减少
     * 如果忘记确认(ACK),则PEL列表会不断增长占用内存
     * 如果服务器发生意外,重启连接后将再次收到PEL中的消息ID列表
     */
    @Bean
    public Subscription subscription(RedisConnectionFactory factory) {
        initGroup("mystream", "group1");
        // 创建Stream消息监听容器配置
        StreamMessageListenerContainer.StreamMessageListenerContainerOptions<String, ObjectRecord<String, String>> options = StreamMessageListenerContainer
                .StreamMessageListenerContainerOptions
                .builder()
                // 读取超时时间
                .pollTimeout(Duration.ofSeconds(3))
                // 配置消息类型
                .targetType(String.class)
                // 异常处理器
                .errorHandler(t -> log.info("redis listener error", t))
                .build();
        // 创建Stream消息监听容器
        StreamMessageListenerContainer<String, ObjectRecord<String, String>> listenerContainer = StreamMessageListenerContainer.create(factory, options);
        // 设置消费手动提交配置
        Subscription subscription = listenerContainer.receive(
                // 设置消费者分组和名称
                Consumer.from("group1","consumer-1"),
                // 设置订阅Stream的key和获取偏移量,以及消费处理类
                StreamOffset.create("mystream", ReadOffset.lastConsumed()),
                agendaListener);
        // 监听容器启动
        listenerContainer.start();
        return subscription;
    }

    /**
     * 初始化分组
     */
    private void initGroup(String key, String group) {
        Boolean aBoolean = redisTemplate.hasKey(key);
        // 创建不存在的分组
        if (Boolean.FALSE.equals(aBoolean)) {
            redisTemplate.opsForStream().createGroup(key, group);
        }
    }

}

实现消息的监听

@Slf4j
@Component
public class MyListener implements StreamListener<String, ObjectRecord<String, String>> {
    @Autowired
    private StringRedisTemplate redisTemplate;

    @Override
    public void onMessage(ObjectRecord<String, String> record) {
        try {
            String value = record.getValue();
            log.info("stream name :{}, body:{}", record.getStream(), value);
            if (StrUtil.isBlank(value)) {
                return;
            }
            // todo 业务逻辑
            // 手动确认消息 如果不ack 消息就会进入到pending队列中 这个队列都是维护消费者的未确认的消息
            redisTemplate.opsForStream().acknowledge("mystream", "group1", record.getId().getValue());
        } catch (Exception e) {
            log.error("error message:{}", e.getMessage());
        }
    }

}

这里说一下消息体类型 Record 官方解释:流中的单个条目,由条目 ID 和实际条目值(通常是字段值对的集合)组成

我们就是可以理解为消息体类型。Record接口,常用的就是

  • MapRecord(键值对类型)

  • ObjectRecord(对象类型)

测试

@PostMapping("/addStream")
public ResponseResult<String> addStream(){
    // 这里的消息体都是string类型
    ObjectRecord<String, String> record = StreamRecords.objectBacked("1234567").withStreamKey("mystream");
    // 这里是消息id,消息id在队列里是唯一的
    RecordId recordId = stringRedisTemplate.opsForStream().add(record);
    // 裁剪队列,因为队列即使被消费者消费后任然不会删除,所以我们队列设定最大容量,也就是上面提到的 XTRIM  命令
    Long count = stringRedisTemplate.opsForStream().trim("mystream", 100000);
    System.out.println("trimCount" + count);
    if (recordId != null) {
        // 返回打印消息id
        return ResponseResult.success(recordId.getValue());
    }
    return ResponseResult.success();
}

基于redisson实现

相关消息监听和消费者配置同上

测试

RStream<Object, Object> stream = redissonClient.getStream("mystream", new SerializationCodec());
StreamAddArgs<Object, Object> entry = StreamAddArgs.entry("a","1");
stream.add(entry);
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,634评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,951评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,427评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,770评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,835评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,799评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,768评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,544评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,979评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,271评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,427评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,121评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,756评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,375评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,579评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,410评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,315评论 2 352

推荐阅读更多精彩内容