场馆预定平台高并发时间段预定实现V2

news2025/1/23 0:31:44

🎯 本文档介绍了场馆预订系统接口V2的设计与实现,旨在解决V1版本中库存数据不一致及性能瓶颈的问题。通过引入令牌机制确保缓存和数据库库存的最终一致性,避免因服务器故障导致的库存错误占用问题。同时,采用消息队列异步处理库存扣减和订单创建,显著提升了接口的吞吐量和响应速度。测试结果显示,新版接口在高并发场景下表现优异,平均响应时间为1801毫秒,吞吐量达到了每秒1045.8次请求,异常率仅为0.22%,极大改善了用户体验。
🏠️ HelloDam/场快订(场馆预定 SaaS 平台)

文章目录

  • 说明
  • 避免空场无法预订
  • 接口性能提升
    • Controller
    • Service
    • MQ
      • 生产者
      • 消费者
  • 测试结果
  • 说明

说明

在阅读此文之前,建议先阅读预订接口V1实现:https://hellodam.blog.csdn.net/article/details/144950335

接口 V2 主要是解决 V1 存在的一些问题:

  • 问题一:接口 V1 中存在如下问题:假如说 lua 脚本执行完成,缓存中的库存已经扣减,结果突然服务器宕机了,没有执行后续的数据库库存扣减和创建订单流程,就会出现库存被错误占用,导致缓存中库存小于实际库存。对应于现实,就是有的场空着,用户预定不到
  • 问题二:接口 V1 中,因为库存扣减和订单创建是同步的,预订接口吞吐量较低。为了进一步提升接口性能,可以使用消息队列来异步执行库存扣减和订单创建逻辑

避免空场无法预订

缓存扣减完成之后,由于发生故障,导致没有更新数据库。这个问题本身是无法避免的,只能通过一些机制来兜底。本文通过使用令牌机制来解决空场无法预订问题。

在接口 V1 中,用户请求预定接口,先查看 Redis 缓存中的库存是否大于 0 ,大于 0 才进行后面的操作。令牌是什么,其实也是这个缓存,但是我们并不完全相信它,我们知道它可能和数据库的数据不一致。当用户获取不到令牌的时候,我们不是直接返回时间段售罄错误,而是先查询一下数据库,看看是不是真的售罄了,如果数据库中还有库存,就删除令牌缓存。这样下一个用户再发起预订时,就会重新刷新令牌缓存,这样令牌的数据就和数据库保持一致,就不会出现空场无法预订的问题。

为了实现这个思路,我们还需要考虑一个问题,难道每个用户看到没有令牌都去查数据库吗,那肯定不行,这样并发高的话,数据库很容易被打崩。可以通过分布式锁让同一时刻只有一个用户查询数据库,但是光是添加分布式锁还是不行,用户请求多时,可能出现不同时间点连续查询数据库刷新token的情况,其实不必如此频繁查询。还有一个问题,高并发时大量任务等着数据库响应,数据库更新不会那么快。如果是立刻刷新token,可能出现数据库没来得及扣减库存,就被刷新到token中了,这样会导致时间段超卖,因为令牌数量大于库存。为了解决上述问题,可以先延时10秒再刷新token,在这10秒内,其他用户访问预定接口,因为拿不到分布式锁,也不会重复执行token刷新。

/**
 * 查询数据库是否还有库存,如果还有的话,删除令牌,让下一个用户重新加载令牌缓存
 *
 * @param timePeriodId
 */
private void refreshTokenByCheckDatabase(Long timePeriodId) {
    RLock lock = redissonClient.getLock(String.format(RedisCacheConstant.VENUE_LOCK_TIME_PERIOD_REFRESH_TOKEN_KEY, timePeriodId));
    // 尝试获取分布式锁,获取不成功直接返回
    if (!lock.tryLock()) {
        return;
    }
    // 延迟 10 秒之后去检查数据库和令牌是否一致
    // 为啥要延迟?如果不延迟的话,可能高峰期时,大量请求过来,数据库还没来得及更新,就触发令牌刷新,导致超卖
    tokenRefreshExecutor.schedule(() -> {
        try {
            TimePeriodDO timePeriodDO = this.getById(timePeriodId);
            if (timePeriodDO.getStock() > 0) {
                // --if-- 数据库中还有库存,说明数据库中的库存和令牌中的库存不一致,删除缓存,让下一个用户重新获取
                stringRedisTemplate.delete(RedisCacheConstant.VENUE_TIME_PERIOD_STOCK_TOKEN_KEY);
                stringRedisTemplate.delete(RedisCacheConstant.VENUE_TIME_PERIOD_FREE_INDEX_BIT_MAP_TOKEN_KEY);
            }
        } finally {
            lock.unlock();
        }
    }, 10, TimeUnit.SECONDS);
}

接口性能提升

Controller

/**
 * 预定时间段
 */
@GetMapping("/v2/reserve")
@Idempotent(
        uniqueKeyPrefix = "vrs-venue:lock_reserve:",
        // 让用户同时最多只能预定一个时间段,根据用户名来加锁
        // key = "T(com.vrs.common.context.UserContext).getUsername()",
        // 让用户同时最多只能预定该时间段一次,但是可以同时预定其他时间段,根据用户名+时间段ID来加锁
        key = "T(com.vrs.common.context.UserContext).getUsername()+'_'+#timePeriodId",
        message = "正在执行场馆预定流程,请勿重复预定...",
        scene = IdempotentSceneEnum.RESTAPI
)
@Operation(summary = "预定时间段V2")
public Result reserve2(@RequestParam("timePeriodId") Long timePeriodId) {
    timePeriodService.reserve2(timePeriodId);
    return Results.success();
}

Service

【预订流程】

  • 参数检验
  • 获取令牌
    • 能获取到,执行下一步
    • 获取不到,查询数据库,刷新令牌缓存
  • 发送消息,异步更新库存并生成订单
/**
 * 尝试获取令牌,令牌获取成功之后,发送消息,异步执行库存扣减和订单生成
 * 注意:令牌在极端情况下,如扣减令牌之后,服务宕机了,此时令牌的库存是小于真实库存的
 * 如果查询令牌发现库存为0,尝试去数据库中加载数据,加载之后库存还是0,说明时间段确实售罄了
 * 使用消息队列异步 扣减库存,更新缓存,生成订单
 *
 * @param timePeriodId
 */
@Override
public void reserve2(Long timePeriodId) {
     参数校验:使用责任链模式校验数据是否正确
    TimePeriodReserveReqDTO timePeriodReserveReqDTO = new TimePeriodReserveReqDTO(timePeriodId);
    chainContext.handler(ChainConstant.RESERVE_CHAIN_NAME, timePeriodReserveReqDTO);
    TimePeriodDO timePeriodDO = timePeriodReserveReqDTO.getTimePeriodDO();
    Long venueId = timePeriodReserveReqDTO.getVenueId();
    VenueDO venueDO = timePeriodReserveReqDTO.getVenueDO();
    PartitionDO partitionDO = partitionService.getPartitionDOById(timePeriodDO.getPartitionId());

     使用lua脚本获取一个空场地对应的索引,并扣除相应的库存,同时在里面进行用户的查重
    // 首先检测空闲场号缓存有没有加载好,没有的话进行加载
    this.checkBitMapCache(
            String.format(RedisCacheConstant.VENUE_TIME_PERIOD_FREE_INDEX_BIT_MAP_TOKEN_KEY, timePeriodReserveReqDTO.getTimePeriodId()),
            timePeriodId,
            partitionDO.getNum());
    // 其次检测时间段库存有没有加载好,没有的话进行加载
    this.getStockByTimePeriodId(RedisCacheConstant.VENUE_TIME_PERIOD_STOCK_TOKEN_KEY, timePeriodReserveReqDTO.getTimePeriodId());
    // 执行lua脚本
    Long freeCourtIndex = executeStockReduceByLua(
            timePeriodReserveReqDTO,
            venueDO,
            RedisCacheConstant.VENUE_TIME_PERIOD_STOCK_TOKEN_KEY,
            RedisCacheConstant.VENUE_TIME_PERIOD_FREE_INDEX_BIT_MAP_TOKEN_KEY);
    if (freeCourtIndex == -2L) {
        // --if-- 用户已经购买过该时间段
        throw new ClientException(BaseErrorCode.TIME_PERIOD_HAVE_BOUGHT_ERROR);
    } else if (freeCourtIndex == -1L) {
        // --if-- 没有空闲的场号,查询数据库,如果数据库中有库存,删除缓存,下一个用户预定时重新加载令牌
        this.refreshTokenByCheckDatabase(timePeriodId);
        throw new ServiceException(BaseErrorCode.TIME_PERIOD_SELL_OUT_ERROR);
    }

     发送消息,异步更新库存并生成订单
    SendResult sendResult = executeReserveProducer.sendMessage(ExecuteReserveMqDTO.builder()
            .timePeriodId(timePeriodId)
            .freeCourtIndex(freeCourtIndex)
            .venueId(venueId)
            .userId(UserContext.getUserId())
            .userName(UserContext.getUsername())
            .build());
    if (!sendResult.getSendStatus().equals(SendStatus.SEND_OK)) {
        log.error("消息发送失败: " + sendResult.getSendStatus());
        // 恢复令牌缓存
        this.restoreStockAndBookedSlotsCache(
                timePeriodId,
                UserContext.getUserId(),
                freeCourtIndex,
                RedisCacheConstant.VENUE_TIME_PERIOD_STOCK_TOKEN_KEY,
                RedisCacheConstant.VENUE_TIME_PERIOD_FREE_INDEX_BIT_MAP_TOKEN_KEY);
        throw new ServiceException(BaseErrorCode.MQ_SEND_ERROR);
    }
}

【获取令牌】

获取令牌的过程其实就是 检验用户是否重新预订、库存数量检查、场号分配、库存扣减、场号占用 ,这里和接口V1的实现是一样的

/**
 * 使用lua脚本,进行缓存中的库存扣减,并分配空闲场号
 *
 * @param timePeriodReserveReqDTO
 * @param venueDO
 * @param stockKey
 * @param freeIndexBitMapKey
 * @return
 */
private Long executeStockReduceByLua(TimePeriodReserveReqDTO timePeriodReserveReqDTO, VenueDO venueDO,
                                     String stockKey, String freeIndexBitMapKey) {
    // 使用 Hutool 的单例管理容器 管理lua脚本的加载,保证其只被加载一次
    String luaScriptPath = "lua/free_court_index_allocate_by_bitmap.lua";
    DefaultRedisScript<Long> luaScript = Singleton.get(luaScriptPath, () -> {
        DefaultRedisScript<Long> redisScript = new DefaultRedisScript<>();
        redisScript.setScriptSource(new ResourceScriptSource(new ClassPathResource(luaScriptPath)));
        redisScript.setResultType(Long.class);
        return redisScript;
    });
    // 执行用户重复预定校验、库存扣减、场号分配
    Long freeCourtIndex = stringRedisTemplate.execute(
            luaScript,
            Lists.newArrayList(
                    String.format(stockKey, timePeriodReserveReqDTO.getTimePeriodId()),
                    String.format(freeIndexBitMapKey, timePeriodReserveReqDTO.getTimePeriodId()),
                    String.format(RedisCacheConstant.VENUE_IS_USER_BOUGHT_TIME_PERIOD_KEY, timePeriodReserveReqDTO.getTimePeriodId())
            ),
            UserContext.getUserId().toString(),
            String.valueOf(venueDO.getAdvanceBookingDay() * 86400)
    );
    return freeCourtIndex;
}

lua

-- 定义脚本参数
local stock_key = KEYS[1]
local free_index_bitmap_key = KEYS[2]
-- 用来存储已购买用户的set
local set_name = KEYS[3]

-- 用户ID
local user_id = ARGV[1]
-- 过期时间 (秒)
local expire_time = tonumber(ARGV[2])

-- 检查用户是否已经购买过
if redis.call("SISMEMBER", set_name, user_id) == 1 then
    -- 用户已经购买过,返回 -2 表示失败
    return -2
end

-- 获取库存
local current_inventory = tonumber(redis.call('GET', stock_key) or 0)

-- 尝试消耗库存
if current_inventory < 1 then
    -- 库存不够了,返回-1,代表分配空场号失败
    return -1 -- 失败
end

-- 查找第一个空闲的场地(位图中第一个为 0 的位)
local free_court_bit = redis.call("BITPOS", free_index_bitmap_key, 0)

if not free_court_bit or free_court_bit == -1 then
    -- 没有空闲的场号
    return -1 -- 失败
end

-- 占用该场地(将对应位设置为 1)
redis.call("SETBIT", free_index_bitmap_key, free_court_bit, 1)
-- 更新库存
redis.call('DECRBY', stock_key, 1)
-- 添加用户到已购买集合
redis.call("SADD", set_name, user_id)
-- 设置过期时间
if expire_time > 0 then
    redis.call("EXPIRE", set_name, expire_time)
end

-- 返回分配的场地索引(注意:位图的位索引从0开始,如果你需要从1开始,这里加1)
return tonumber(free_court_bit)

【更新缓存中库存】

大家可能会疑问,为啥有了令牌,还要更新缓存中的库存和空闲场号。因为我们在前端展示的信息需要是真实的库存信息,为了加速查询,需要将库存缓存起来,这里的缓存数据需要和数据库一致。为了保证缓存和数据库的最终一致性,可以开启 binlog ,然后使用 Canal 进行监听。如果数据库中的数据更新了,就发送消息到消息队列中,消费消息时再更新缓存中的库存。

-- 定义脚本参数
local stock_key = KEYS[1]
local free_index_bitmap_key = KEYS[2]

-- 预订场号
local free_court_bit = ARGV[1]

-- 占用该场地(将对应位设置为 1)
redis.call("SETBIT", free_index_bitmap_key, free_court_bit, 1)
-- 更新库存
redis.call('DECRBY', stock_key, 1)

return 0

【检测和加载位图缓存】

/**
 * 检测位图缓存是否加载好,没有的话,执行加载操作
 *
 * @param freeIndexBitmapKey
 * @param timePeriodId
 * @param initStock
 */
@Override
public void checkBitMapCache(String freeIndexBitmapKey, Long timePeriodId, int initStock) {
    String cache = stringRedisTemplate.opsForValue().get(freeIndexBitmapKey);
    if (StringUtils.isBlank(cache)) {
        // --if-- 如果缓存中的位图为空
        RLock lock = redissonClient.getLock(String.format(RedisCacheConstant.VENUE_LOCK_TIME_PERIOD_FREE_INDEX_BIT_MAP_KEY, timePeriodId));
        lock.lock();
        try {
            // 双重判定一下,避免其他线程已经加载数据到缓存中了
            cache = stringRedisTemplate.opsForValue().get(freeIndexBitmapKey);
            if (StringUtils.isBlank(cache)) {
                // --if-- 如果缓存中的位图还是空,到数据库中加载位图
                TimePeriodDO timePeriodDO = this.getById(timePeriodId);
                if (timePeriodDO == null) {
                    throw new ServiceException(timePeriodId + "对应的时间段为null", BaseErrorCode.SERVICE_ERROR);
                }
                // 将位图信息设置到缓存中
                this.initializeFreeIndexBitmap(freeIndexBitmapKey, initStock, timePeriodDO.getBookedSlots(), 24 * 3600);
            }
        } finally {
            // 解锁
            lock.unlock();
        }
    }
}

/**
 * 初始化Redis中的位图,并设置key的过期时间
 *
 * @param freeIndexBitmapKey 位图的键名
 * @param longValue          用于初始化位图的 long 类型数据
 * @param expireSecond       key的过期时间(秒)
 */
public void initializeFreeIndexBitmap(String freeIndexBitmapKey, int initStock, long longValue, long expireSecond) {
    // 将 long 转换为64位的二进制字符串
    String binaryString = Long.toBinaryString(longValue);
    // 确保字符串长度为64位,不足的部分用0补齐
    binaryString = String.format("%64s", binaryString).replace(' ', '0');

    // 从低位到高位遍历二进制字符串,设置位图中的对应位
    for (int i = 0; i < 64 && initStock-- >= 0; i++) {
        // 注意:long的最低位对应位图的第0位
        if (binaryString.charAt(63 - i) == '1') {
            stringRedisTemplate.opsForValue().setBit(freeIndexBitmapKey, i, true).booleanValue();
        } else {
            stringRedisTemplate.opsForValue().setBit(freeIndexBitmapKey, i, false).booleanValue();
        }
    }

    // 设置过期时间,仅当expireTime大于0时进行设置
    if (expireSecond > 0) {
        stringRedisTemplate.expire(freeIndexBitmapKey, expireSecond, TimeUnit.SECONDS);
    }
}

【检验和加载库存缓存】

这里使用了封装的缓存组件,需要去仓库查看详细代码

/**
 * 获取指定时间段的库存
 *
 * @param timePeriodId
 * @return
 */
@Override
public Integer getStockByTimePeriodId(Long timePeriodId) {
    return (Integer) distributedCache.safeGet(
            String.format(RedisCacheConstant.VENUE_TIME_PERIOD_STOCK_KEY, timePeriodId),
            new TypeReference<Integer>() {
            },
            () -> {
                TimePeriodDO timePeriodDO = this.getById(timePeriodId);
                return timePeriodDO.getStock();
            },
            1,
            TimeUnit.DAYS);
}

【消费消息,执行预订流程】

和接口 V1 不同的是,V1 时同步创建订单,创建完成之后,直接访问给用户订单数据。但是在 V2 中,将任务交给消息队列之后,就要返回成功了。用户需要在前端等待订单创建结果。那前端如何感知订单是否创建成功呢?

  • 方式一:前端轮询查询后端,如每隔一秒问一下后端,订单创建好没有,创建好了就返回给前端,这样前端就可以进行支付了
  • 方式二:使用前后端双向通讯技术,如WebSocket。前后端一开始先建立好连接,等后端消费消息,创建订单成功之后,直接将订单信息推送给前端
/**
 * 通过消息队列执行 时间段预定 逻辑
 * @param executeReserveMqDTO
 */
@Override
public void mqExecutePreserve(ExecuteReserveMqDTO executeReserveMqDTO) {
    TimePeriodDO timePeriodDO = this.getTimePeriodDOById(executeReserveMqDTO.getTimePeriodId());
    // 编程式开启事务,减少事务粒度,避免长事务的发生
    transactionTemplate.executeWithoutResult(status -> {
        try {
            // 扣减当前时间段的库存,修改空闲场信息
            baseMapper.updateStockAndBookedSlots(timePeriodDO.getId(), timePeriodDO.getPartitionId(), executeReserveMqDTO.getFreeCourtIndex());
            // 更新缓存中的库存、位图
            if (!isUseBinlog) {
                // --if-- 如果不使用binlog,需要手动更新缓存

                // 首先检测空闲场号缓存有没有加载好,没有的话进行加载
                this.checkBitMapCache(
                        String.format(RedisCacheConstant.VENUE_TIME_PERIOD_FREE_INDEX_BIT_MAP_KEY, executeReserveMqDTO.getTimePeriodId()),
                        executeReserveMqDTO.getTimePeriodId(),
                        partitionService.getPartitionDOById(timePeriodDO.getPartitionId()).getNum());
                // 其次检测时间段库存有没有加载好,没有的话进行加载
                this.getStockByTimePeriodId(executeReserveMqDTO.getTimePeriodId());
                // 使用 Hutool 的单例管理容器 管理lua脚本的加载,保证其只被加载一次
                String luaScriptPath = "lua/inventory_update.lua";
                DefaultRedisScript<Long> luaScript = Singleton.get(luaScriptPath, () -> {
                    DefaultRedisScript<Long> redisScript = new DefaultRedisScript<>();
                    redisScript.setScriptSource(new ResourceScriptSource(new ClassPathResource(luaScriptPath)));
                    redisScript.setResultType(Long.class);
                    return redisScript;
                });
                // 库存扣减、场号占用
                stringRedisTemplate.execute(
                        luaScript,
                        Lists.newArrayList(
                                String.format(RedisCacheConstant.VENUE_TIME_PERIOD_STOCK_KEY, executeReserveMqDTO.getTimePeriodId()),
                                String.format(RedisCacheConstant.VENUE_TIME_PERIOD_FREE_INDEX_BIT_MAP_KEY, executeReserveMqDTO.getTimePeriodId())
                        ),
                        executeReserveMqDTO.getFreeCourtIndex().toString()
                );
            }
            // todo 需要实现binlog版本

            // 调用远程服务创建订单
            OrderGenerateReqDTO orderGenerateReqDTO = OrderGenerateReqDTO.builder()
                    .timePeriodId(timePeriodDO.getId())
                    .partitionId(timePeriodDO.getPartitionId())
                    .periodDate(timePeriodDO.getPeriodDate())
                    .beginTime(timePeriodDO.getBeginTime())
                    .endTime(timePeriodDO.getEndTime())
                    .courtIndex(executeReserveMqDTO.getFreeCourtIndex())
                    .userId(executeReserveMqDTO.getUserId())
                    .userName(executeReserveMqDTO.getUserName())
                    .venueId(executeReserveMqDTO.getVenueId())
                    .payAmount(timePeriodDO.getPrice())
                    .build();

            Result<OrderDO> result;
            try {
                result = orderFeignService.generateOrder(orderGenerateReqDTO);
                if (result == null || !result.isSuccess()) {
                    // --if-- 订单生成失败,抛出异常,上面的库存扣减也会回退
                    throw new ServiceException(BaseErrorCode.ORDER_GENERATE_ERROR);
                }
            } catch (Exception e) {
                // --if-- 订单生成服务调用失败
                // 恢复缓存中的信息
                this.restoreStockAndBookedSlotsCache(
                        timePeriodDO.getId(),
                        1L,
                        executeReserveMqDTO.getFreeCourtIndex(),
                        RedisCacheConstant.VENUE_TIME_PERIOD_STOCK_TOKEN_KEY,
                        RedisCacheConstant.VENUE_TIME_PERIOD_FREE_INDEX_BIT_MAP_TOKEN_KEY);
                // todo 如果说由于网络原因,实际上订单已经创建成功了,但是因为超时访问失败,这里库存却回滚了,此时需要将订单置为废弃状态(即删除)
                // 发送一个短暂的延时消息(时间过长,用户可能已经支付),去检查订单是否生成,如果生成,将其删除
                // 打印错误堆栈信息
                e.printStackTrace();
                // 把错误返回到前端
                throw new ServiceException(e.getMessage());
            }
            OrderDO orderDO = result.getData();
            // todo 使用 WebSocket 通知前端,订单生成成功
        } catch (Exception ex) {
            status.setRollbackOnly();
            throw ex;
        }
    });
}

MQ

生产者

import cn.hutool.core.util.StrUtil;
import com.vrs.constant.RocketMqConstant;
import com.vrs.domain.dto.mq.ExecuteReserveMqDTO;
import com.vrs.templateMethod.AbstractCommonSendProduceTemplate;
import com.vrs.templateMethod.BaseSendExtendDTO;
import com.vrs.templateMethod.MessageWrapper;
import lombok.extern.slf4j.Slf4j;
import org.apache.rocketmq.common.message.MessageConst;
import org.springframework.messaging.Message;
import org.springframework.messaging.support.MessageBuilder;
import org.springframework.stereotype.Component;

import java.util.UUID;

/**
 * 执行预订流程 生产者
 *
 * @Author dam
 * @create 2024/9/20 16:00
 */
@Slf4j
@Component
public class ExecuteReserveProducer extends AbstractCommonSendProduceTemplate<ExecuteReserveMqDTO> {

    @Override
    protected BaseSendExtendDTO buildBaseSendExtendParam(ExecuteReserveMqDTO messageSendEvent) {
        return BaseSendExtendDTO.builder()
                .eventName("执行时间段预定")
                .keys(String.valueOf(messageSendEvent.getTimePeriodId()))
                .topic(RocketMqConstant.VENUE_TOPIC)
                .tag(RocketMqConstant.TIME_PERIOD_EXECUTE_RESERVE_TAG)
                .sentTimeout(2000L)
                .build();
    }

    @Override
    protected Message<?> buildMessage(ExecuteReserveMqDTO messageSendEvent, BaseSendExtendDTO requestParam) {
        String keys = StrUtil.isEmpty(requestParam.getKeys()) ? UUID.randomUUID().toString() : requestParam.getKeys();
        return MessageBuilder
                .withPayload(new MessageWrapper(keys, messageSendEvent))
                .setHeader(MessageConst.PROPERTY_KEYS, keys)
                .setHeader(MessageConst.PROPERTY_TAGS, requestParam.getTag())
                .build();
    }
}

消费者

import com.vrs.annotation.Idempotent;
import com.vrs.constant.RocketMqConstant;
import com.vrs.domain.dto.mq.ExecuteReserveMqDTO;
import com.vrs.enums.IdempotentSceneEnum;
import com.vrs.service.TimePeriodService;
import com.vrs.templateMethod.MessageWrapper;
import lombok.RequiredArgsConstructor;
import lombok.SneakyThrows;
import lombok.extern.slf4j.Slf4j;
import org.apache.rocketmq.spring.annotation.MessageModel;
import org.apache.rocketmq.spring.annotation.RocketMQMessageListener;
import org.apache.rocketmq.spring.annotation.SelectorType;
import org.apache.rocketmq.spring.core.RocketMQListener;
import org.springframework.stereotype.Component;

/**
 * 执行预订流程 消费者
 * @Author dam
 * @create 2024/9/20 21:30
 */
@Slf4j(topic = RocketMqConstant.VENUE_TOPIC)
@Component
@RocketMQMessageListener(topic = RocketMqConstant.VENUE_TOPIC,
        consumerGroup = RocketMqConstant.VENUE_CONSUMER_GROUP + "-" + RocketMqConstant.TIME_PERIOD_EXECUTE_RESERVE_TAG,
        messageModel = MessageModel.CLUSTERING,
        // 监听tag
        selectorType = SelectorType.TAG,
        selectorExpression = RocketMqConstant.TIME_PERIOD_EXECUTE_RESERVE_TAG
)
@RequiredArgsConstructor
public class ExecuteReserveListener implements RocketMQListener<MessageWrapper<ExecuteReserveMqDTO>> {

    private final TimePeriodService timePeriodService;

    /**
     * 消费消息的方法
     * 方法报错就会拒收消息
     *
     * @param messageWrapper 消息内容,类型和上面的泛型一致。如果泛型指定了固定的类型,消息体就是我们的参数
     */
    @Idempotent(
            uniqueKeyPrefix = "time_period_execute_reserve:",
            key = "#messageWrapper.getMessage().getTimePeriodId()+''",
            scene = IdempotentSceneEnum.MQ,
            keyTimeout = 3600L
    )
    @SneakyThrows
    @Override
    public void onMessage(MessageWrapper<ExecuteReserveMqDTO> messageWrapper) {
        // 开头打印日志,平常可 Debug 看任务参数,线上可报平安(比如消息是否消费,重新投递时获取参数等)
        log.info("[消费者] 执行时间段预定,时间段ID:{}", messageWrapper.getMessage().getTimePeriodId());
        timePeriodService.mqExecutePreserve(messageWrapper.getMessage());
    }
}

测试结果

在这里插入图片描述

  1. 样本数量:共有40,000个样本,这表示在测试期间进行了40,000次操作或请求。
  2. 响应时间
    1. 平均值:1801毫秒,表示所有请求的平均响应时间。
    2. 中位数:1346毫秒,表示50%的请求响应时间低于这个值。
    3. 90%百分位:2048毫秒,表示90%的请求响应时间低于这个值。
    4. 95%百分位:3410毫秒,表示95%的请求响应时间低于这个值。
    5. 99%百分位:15133毫秒,表示99%的请求响应时间低于这个值。
    6. 最小值:15毫秒,表示最快的请求响应时间。
    7. 最大值:22121毫秒,表示最慢的请求响应时间。
  3. 异常率:0.22%,表示在所有请求中,有0.22%的请求出现了异常。
  4. 吞吐量:每秒可以处理1045.8个请求
  5. 网络流量
    1. 接收速率:221.51 KB/sec,表示系统每秒接收的数据量。
    2. 发送速率:509.96 KB/sec,表示系统每秒发送的数据量。

总结

  • 系统的平均响应时间为1801毫秒,中位数为1346毫秒,表明大多数请求的响应时间在可接受范围内。
  • 99%的请求响应时间在15133毫秒以内,但有少数请求的响应时间较长,最大值达到了22121毫秒。
  • 系统的吞吐量为1045.8次请求/秒,处理能力较高,相较于接口V1,性能强了一倍

说明

文章内容并非最新代码实现,若需要知道最新实现,麻烦移步开源仓库: HelloDam/场快订(场馆预定 SaaS 平台)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2280620.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

图解Git——分布式Git《Pro Git》

分布式工作流程 Centralized Workflow&#xff08;集中式工作流&#xff09; 所有开发者都与同一个中央仓库同步代码&#xff0c;每个人通过拉取、提交来合作。如果两个开发者同时修改了相同的文件&#xff0c;后一个开发者必须在推送之前合并其他人的更改。 Integration-Mana…

将 AzureBlob 的日志通过 Azure Event Hubs 发给 Elasticsearch(1.标准版)

问题 项目里使用了 AzureBlob 存储了用户上传的各种资源文件&#xff0c;近期 AzureBlob 的流量费用增长很快&#xff0c;想通过分析Blob的日志&#xff0c;获取一些可用的信息&#xff0c;所以有了这个需求&#xff1a;将存储账户的日志&#xff08;读写&#xff0c;审计&…

【json_object】mysql中json_object函数过长,显示不全

问题&#xff1a;json只显示部分 解决&#xff1a; SET GLOBAL group_concat_max_len 1000000; -- 设置为1MB&#xff0c;根据需要调整如果当前在navicat上修改&#xff0c;只有效本次连接和后续会话&#xff0c;重新连接还是会恢复默认值1024 在my.ini配置文件中新增或者修…

Flink底层架构与运行流程

这张图展示了Flink程序的架构和运行流程。 主要组件及功能&#xff1a; Flink Program&#xff08;Flink程序&#xff09;&#xff1a; 包含Program code&#xff08;程序代码&#xff09;&#xff0c;这是用户编写的业务逻辑代码。经过Optimizer / Graph Builder&#xff08…

你还在用idea吗

从VIM、Emacs&#xff0c;到eclipse、Jetbrains, 再到VSCode&#xff0c;过去的三十年时间&#xff0c;出现了这三代IDE产品。现在属于AI的时代来了&#xff0c;最新一代的产品像Cursor、Windsurf&#xff0c;就在昨天&#xff0c;字节跳动发布了最新的IDE&#xff0c;就叫Trae…

Linux TCP 之 RTT 采集与 RTO 计算

我们来看看 Linux TCP 采集 RTT 的函数 tcp_rtt_estimator&#xff0c;看注释&#xff0c;充满了胶着。 但在那个谨慎的年代&#xff0c;这些意味着什么&#xff1f; RTT 最初仅用于 RTO 的计算而不是用于调速&#xff0c;RTO 的计算存在两个问题&#xff0c;如果过估&#x…

学习ASP.NET Core的身份认证(基于JwtBearer的身份认证9)

测试数据库中只有之前记录温湿度及烟雾值的表中数据较多&#xff0c;在该数据库中增加AppUser表&#xff0c;用于登录用户身份查询&#xff0c;数据库表如下所示&#xff1a;   项目中安装SqlSugarCore包&#xff0c;然后修改控制器类的登录函数及分页查询数据函数&#xff…

leetcode-75-颜色分类

文章目录 1. 归并排序2. 计数3. 按照题目要求&#xff0c;原地腾挪 给定一个包含红色、白色和蓝色、共 n 个元素的数组 nums &#xff0c;原地 对它们进行排序&#xff0c;使得相同颜色的元素相邻&#xff0c;并按照红色、白色、蓝色顺序排列。 我们使用整数 0、 1 和 2 分别表…

html,css,js的粒子效果

这段代码实现了一个基于HTML5 Canvas的高级粒子效果&#xff0c;用户可以通过鼠标与粒子进行交互。下面是对代码的详细解析&#xff1a; HTML部分 使用<!DOCTYPE html>声明文档类型。<html>标签内包含了整个网页的内容。<head>部分定义了网页的标题&#x…

学习记录之原型,原型链

构造函数创建对象 Person和普通函数没有区别&#xff0c;之所以是构造函数在于它是通过new关键字调用的&#xff0c;p就是通过构造函数Person创建的实列对象 function Person(age, name) {this.age age;this.name name;}let p new Person(18, 张三);prototype prototype n…

go语言zero框架通过chromedp实现网页在线截图的设计与功能实现

在 GoZero 框架中实现网页在线截图的功能&#xff0c;可以通过集成 chromedp 库来控制 Chrome 浏览器进行截图。chromedp 是一个基于 Chrome DevTools 协议的 Go 包&#xff0c;可以用来在 Go 程序中模拟浏览器操作&#xff0c;如页面截图、DOM 操作、表单提交等。 下面是一个…

以 RFID 为钥,开启民兵装备管理的科技之门

民兵配备的武器及装备涵盖了各式武器、弹药及军事技术设备&#xff0c;其管理的优良直接决定了民兵的作战效能。鉴于民兵装备普遍面临老化、维护支援不充分等问题&#xff0c;迫切需要迅速建立完善的民兵装备管理新体系。这一转变将推动民兵装备由数量扩张转向质量提升&#xf…

电脑办公技巧之如何在 Word 文档中添加文字或图片水印

Microsoft Word是全球最广泛使用的文字处理软件之一&#xff0c;它为用户提供了丰富的编辑功能来美化和保护文档。其中&#xff0c;“水印”是一种特别有用的功能&#xff0c;它可以用于标识文档状态&#xff08;如“草稿”或“机密”&#xff09;、公司标志或是版权信息等。本…

llama-factory实战: 基于qwen2.5-7b 手把手实战 自定义数据集清洗 微调

基于qwen2.5 手把手实战 自定义数据集 微调&#xff08;llama-factory&#xff09; 准备工作1.数据集准备&#xff08;例:民法典.txt&#xff09;2.服务器准备&#xff08;阿里云 DSW 白嫖&#xff09;3.环境配置pip 升级模型下载微调助手 4.数据集处理脚本文件4.1文本分割(ber…

微透镜阵列精准全检,白光干涉3D自动量测方案提效70%

广泛应用的微透镜阵列 微透镜是一种常见的微光学元件&#xff0c;通过设计微透镜&#xff0c;可对入射光进行扩散、光束整形、光线均分、光学聚焦、集成成像等调制&#xff0c;进而实现许多传统光学元器件难以实现的特殊功能。 微透镜阵列&#xff08;Microlens Array&#x…

详解单片机学的是什么?(电子硬件)

大家好&#xff0c;我是山羊君Goat。 单片机&#xff0c;对于每一个硬件行业的从业者或者在校电子类专业的学生&#xff0c;相信对于这个名词都不陌生&#xff0c;但是掌没掌握就另说了。 那单片机到底学的是什么呢&#xff1f; 其实单片机在生活中就非常常见&#xff0c;目前…

基于Docker的Spark分布式集群

目录 1. 说明 2. 服务器规划 3. 步骤 3.1 要点 3.2 配置文件 3.2 访问Spark Master 4. 使用测试 5. 参考 1. 说明 以docker容器方式实现apache spark计算集群&#xff0c;能灵活的增减配置与worker数目。 2. 服务器规划 服务器 (1master, 3workers) ip开放端口备注ce…

9. 神经网络(一.神经元模型)

首先&#xff0c;先看一个简化的生物神经元结构&#xff1a; 生物神经元有多种类型&#xff0c;内部也有复杂的结构&#xff0c;但是可以把单个神经元简化为3部分组成&#xff1a; 树突&#xff1a;一个神经元往往有多个树突&#xff0c;用于接收传入的信息。轴突&#xff1a;…

web-view环境下,H5页面打开其他小程序

在Web-view环境下&#xff0c;H5页面无法直接打开其他小程序。正确的实现方式是先从H5页面跳转回当前小程序&#xff0c;再由当前小程序跳转到目标小程序。具体实现方法如下&#xff1a; H5页面跳转回小程序时&#xff0c;调用wx.miniProgram.navigateTo()方法。 小程序跳转到…

数据恢复常用方法(三)如何辨别固态硬盘故障类型

数据恢复首先需要辨别固态硬盘故障类型&#xff0c;只有先确认故障类型&#xff0c;才能进行下一步动作 如下是一种常见的场景&#xff0c;固态硬盘无法识别&#xff0c;接入电源与数据线&#xff0c;电脑的磁盘管理不显示任何信息。 第一步&#xff1a;确认硬件状态&#xff…