【CouponHub项目开发】使用RocketMQ5.x实现延时修改优惠券状态,并通过使用模板方法模式重构消息队列发送功能

在上个章节中我实现了创建优惠券模板的功能,但是,优惠券总会有过期时间,我们怎么去解决到期自动修改优惠券状态这样一个功能呢?我们可以使用RocketMQ5.x新出的任意定时发送消息功能来解决。

初始方案:

  • 首先在创建优惠券模板下面继续添加代码。首先先创建topic
  • 接着去定义消息体
java 复制代码
// 执行 RocketMQ5.x 消息队列发送&异常处理逻辑
    SendResult sendResult;
    try {
        sendResult = rocketMQTemplate.syncSendDeliverTimeMills(couponTemplateDelayCloseTopic, message, deliverTimeStamp);
        log.info("[生产者] 优惠券模板延时关闭 - 发送结果:{},消息ID:{},消息Keys:{}", sendResult.getSendStatus(), sendResult.getMsgId(), messageKeys);
    } catch (Exception ex) {
        log.error("[生产者] 优惠券模板延时关闭 - 消息发送失败,消息体:{}", couponTemplateDO.getId(), ex);
    }
  • 接着去调用相关方法去发送消息

这样一个延时发送的消息生产者就定义好了,接着我们去定义消费者。

  • 我们可以创建一个消费者类让他去实现RocketMQListener然后添加 @RocketMQMessageListener 注解,其中加上 Topic 和消费者组定义
  • 接着我们可以在onMessage方法内去定义我们接收到消息以后需要去做什么
  • 这里我们需要去修改数据库当中的优惠券的状态

这样就已经解决了定时修改优惠券模板状态这样的功能,但是这样直接写在代码当中不是太优雅,而且耦合度太高。我们可以通过去定义一个生产者的抽象类(模板),然后去通过定义具体的类来继承这些抽象类。然后在创建优惠券模板的方法中我们只需要去定义一个事件,然后调用该类的方法去发送消息即可。

java 复制代码
@RequiredArgsConstructor
@Slf4j(topic = "CommonSendProduceTemplate")
public abstract class AbstractCommonSendProduceTemplate<T> {

    private final RocketMQTemplate rocketMQTemplate;

    /**
     * 构建消息发送事件基础扩充属性实体
     *
     * @param messageSendEvent 消息发送事件
     * @return 扩充属性实体
     */
    protected abstract BaseSendExtendDTO buildBaseSendExtendParam(T messageSendEvent);

    /**
     * 构建消息基本参数,请求头、Keys...
     *
     * @param messageSendEvent 消息发送事件
     * @param requestParam     扩充属性实体
     * @return 消息基本参数
     */
    protected abstract Message<?> buildMessage(T messageSendEvent, BaseSendExtendDTO requestParam);

    /**
     * 消息事件通用发送
     *
     * @param messageSendEvent 消息发送事件
     * @return 消息发送返回结果
     */
    public SendResult sendMessage(T messageSendEvent) {
        BaseSendExtendDTO baseSendExtendDTO = buildBaseSendExtendParam(messageSendEvent);
        SendResult sendResult;
        try {
            // 构建 Topic 目标落点 formats: `topicName:tags`
            StringBuilder destinationBuilder = StrUtil.builder().append(baseSendExtendDTO.getTopic());
            if (StrUtil.isNotBlank(baseSendExtendDTO.getTag())) {
                destinationBuilder.append(":").append(baseSendExtendDTO.getTag());
            }

            // 延迟时间不为空,发送任意延迟消息,否则发送普通消息
            if (baseSendExtendDTO.getDelayTime() != null) {
                sendResult = rocketMQTemplate.syncSendDeliverTimeMills(
                        destinationBuilder.toString(),
                        buildMessage(messageSendEvent, baseSendExtendDTO),
                        baseSendExtendDTO.getDelayTime()
                );
            } else {
                sendResult = rocketMQTemplate.syncSend(
                        destinationBuilder.toString(),
                        buildMessage(messageSendEvent, baseSendExtendDTO),
                        baseSendExtendDTO.getSentTimeout()
                );
            }

            log.info("[生产者] {} - 发送结果:{},消息ID:{},消息Keys:{}", baseSendExtendDTO.getEventName(), sendResult.getSendStatus(), sendResult.getMsgId(), baseSendExtendDTO.getKeys());
        } catch (Throwable ex) {
            log.error("[生产者] {} - 消息发送失败,消息体:{}", baseSendExtendDTO.getEventName(), JSON.toJSONString(messageSendEvent), ex);
            throw ex;
        }

        return sendResult;
    }
}

在该抽象类中定义了两个抽象方法,

  • protected abstract BaseSendExtendDTO buildBaseSendExtendParam(T messageSendEvent);该方法用来补充事件的相关属性

  • protected abstract Message<?> buildMessage(T messageSendEvent, BaseSendExtendDTO requestParam);该方法是用来补充消息的属性

  • sendMessage方法中首先会通过buildBaseSendExtendParam方法把事件的属性进行补充,比如延时时间等。

  • 然后会获取这个时间是否存在延时时间这个属性,如果有则发送延时消息,如果没有,就发送普通消息。

这就是延时发送消息的生产者的一个实现类

这是消费者没有太大变化。

我们现在梳理一下它的执行流程

  • 创建完优惠券模板之后,我们根据优惠券的相关信息构建出一个发送事件templateDelayEvent
  • 然后通过自动注入的couponTemplateDelayExecuteStatusProducer延时发送消息的生产者,调用它的发送消息方法进行延时发送消息。
  • 这个生产者是去实现了生产者的一个抽象类,并实现了补充事件属性,以及构建消息体的抽象方法。
  • 然后消费者通过监听该消息来完成消费。
相关推荐
杨杨杨大侠3 小时前
实战案例:商品详情页数据聚合服务的技术实现
java·spring·github
杨杨杨大侠3 小时前
实战案例:保险理赔线上审核系统的技术实现
java·spring·github
小蜗牛在漫步3 小时前
23种设计模式-模板方法模式
设计模式·模板方法模式
计算机毕设定制辅导-无忧学长3 小时前
MQTT 与 Java 框架集成:Spring Boot 实战(一)
java·网络·spring boot
叫我阿柒啊3 小时前
从Java全栈到Vue3实战:一次真实面试的深度复盘
java·spring boot·微服务·vue3·响应式编程·前后端分离·restful api
快乐非自愿3 小时前
掌握设计模式--模板方法模式
java·设计模式·模板方法模式
云飞云共享云桌面3 小时前
SolidWorks对电脑的硬件配置要求具体有哪些
java·服务器·前端·网络·数据库
塔子终结者4 小时前
网络安全A模块专项练习任务十解析
java·服务器·网络安全
泉城老铁4 小时前
Spring Boot中实现多线程分片下载
java·spring boot·后端