解决高并发下购票压力的终极杀招无锁化
提要
在 业务讲解-如何应对高并发下的购票压力 文章中讲解了用户流程,关于如何加分布式锁,和修改数据进行生成订单的流程
在 业务讲解-如何对锁进行优化更好的缓解购票压力 文章中有讲解了如何拆分分布式锁的粒度、引入本地锁的方案来对锁进行优化
如果小伙伴还没有看过,一定要去先学习,然后再来阅读本文
分析
我们再回顾一下订单生成的逻辑
service层
com.damai.service.ProgramOrderService#create
public String create(ProgramOrderCreateDto programOrderCreateDto) {
ProgramShowTime programShowTime =
programShowTimeService.selectProgramShowTimeByProgramIdMultipleCache(programOrderCreateDto.getProgramId());
List<TicketCategoryVo> getTicketCategoryList =
getTicketCategoryList(programOrderCreateDto,programShowTime.getShowTime());
BigDecimal parameterOrderPrice = new BigDecimal("0");
BigDecimal databaseOrderPrice = new BigDecimal("0");
List<SeatVo> purchaseSeatList = new ArrayList<>();
List<SeatDto> seatDtoList = programOrderCreateDto.getSeatDtoList();
List<SeatVo> seatVoList = new ArrayList<>();
Map<String, Long> ticketCategoryRemainNumber = new HashMap<>(16);
for (TicketCategoryVo ticketCategory : getTicketCategoryList) {
List<SeatVo> allSeatVoList =
seatService.selectSeatResolution(programOrderCreateDto.getProgramId(), ticketCategory.getId(),
DateUtils.countBetweenSecond(DateUtils.now(), programShowTime.getShowTime()), TimeUnit.SECONDS);
seatVoList.addAll(allSeatVoList.stream().
filter(seatVo -> seatVo.getSellStatus().equals(SellStatus.NO_SOLD.getCode())).toList());
ticketCategoryRemainNumber.putAll(ticketCategoryService.getRedisRemainNumberResolution(
programOrderCreateDto.getProgramId(),ticketCategory.getId()));
}
if (CollectionUtil.isNotEmpty(seatDtoList)) {
Map<Long, Long> seatTicketCategoryDtoCount = seatDtoList.stream()
.collect(Collectors.groupingBy(SeatDto::getTicketCategoryId, Collectors.counting()));
for (Entry<Long, Long> entry : seatTicketCategoryDtoCount.entrySet()) {
Long ticketCategoryId = entry.getKey();
Long purchaseCount = entry.getValue();
Long remainNumber = Optional.ofNullable(ticketCategoryRemainNumber.get(String.valueOf(ticketCategoryId)))
.orElseThrow(() -> new DaMaiFrameException(BaseCode.TICKET_CATEGORY_NOT_EXIST_V2));
if (purchaseCount > remainNumber) {
throw new DaMaiFrameException(BaseCode.TICKET_REMAIN_NUMBER_NOT_SUFFICIENT);
}
}
Map<String, SeatVo> seatVoMap = seatVoList.stream().collect(Collectors
.toMap(seat -> seat.getRowCode() + "-" + seat.getColCode(), seat -> seat, (v1, v2) -> v2));
for (SeatDto seatDto : seatDtoList) {
SeatVo seatVo = seatVoMap.get(seatDto.getRowCode() + "-" + seatDto.getColCode());
if (Objects.isNull(seatVo)) {
throw new DaMaiFrameException(BaseCode.SEAT_IS_NOT_NOT_SOLD);
}
purchaseSeatList.add(seatVo);
parameterOrderPrice = parameterOrderPrice.add(seatDto.getPrice());
databaseOrderPrice = databaseOrderPrice.add(seatVo.getPrice());
}
if (parameterOrderPrice.compareTo(databaseOrderPrice) > 0) {
throw new DaMaiFrameException(BaseCode.PRICE_ERROR);
}
}else {
Long ticketCategoryId = programOrderCreateDto.getTicketCategoryId();
Integer ticketCount = programOrderCreateDto.getTicketCount();
Long remainNumber = Optional.ofNullable(ticketCategoryRemainNumber.get(String.valueOf(ticketCategoryId)))
.orElseThrow(() -> new DaMaiFrameException(BaseCode.TICKET_CATEGORY_NOT_EXIST_V2));
if (ticketCount > remainNumber) {
throw new DaMaiFrameException(BaseCode.TICKET_REMAIN_NUMBER_NOT_SUFFICIENT);
}
purchaseSeatList = SeatMatch.findAdjacentSeatVos(seatVoList.stream().filter(seatVo ->
Objects.equals(seatVo.getTicketCategoryId(), ticketCategoryId)).collect(Collectors.toList()), ticketCount);
if (purchaseSeatList.size() < ticketCount) {
throw new DaMaiFrameException(BaseCode.SEAT_OCCUPY);
}
}
updateProgramCacheDataResolution(programOrderCreateDto.getProgramId(),purchaseSeatList,OrderStatus.NO_PAY);
return doCreate(programOrderCreateDto,purchaseSeatList);
}
付费内容提示
该文档的全部内容仅对「JavaUp项目实战&技术讲解」知识星球用户开放
加入星球后,你可以获得:
- 超级八股文:100万+字的全栈技术知识库,涵盖技术核心、数据库、中间件、分布式等深度剖析的讲解
- 讲解文档:黑马点评Plus、大麦、大麦pro、大麦AI、流量切换、数据中台的从0到1的550+详细文档
- 讲解视频:黑马点评Plus、大麦、大麦pro、大麦AI、流量切换、数据中台的核心业务详细讲解
- 1 对 1 解答:可以对我进行1对1的问题提问,而不仅仅只限于项目
- 针对性服务:有没理解的地方,文档或者视频还没有讲到可以提出,本人会补充
- 面试与简历指导:提供面试回答技巧,项目怎样写才能在简历中具有独特的亮点
- 中间件环境:对于项目中需要使用的中间件,可直接替换成我提供的云环境
- 面试后复盘:小伙伴去面试后,如果哪里被面试官问住了,可以再找我解答
- 远程的解决:如果在启动项目遇到问题,本人可以帮你远程解决
进入星球后,即可享受上述所有服务,保证不会再有其他隐藏费用。
