• 防止重复下单(redis+数据库唯一索引requestId实现幂等)


    为什么会重复下单

    为什么会重复下单,对于订单服务而言,就是接到了多个下单的请求,原因可能有很多,最常见的是这两种:

    • 用户重复提交
    • 网络原因导致的超时重试
      在这里插入图片描述

    如何防止重复下单

    防止用户提交,最常规的做法,就是客户端 点击下单之后,在 收到服务端响应之前,按钮置灰!!!

    当然,防止重复下单,肯定不能只依靠客户端,可能会因为一些网络的抖动导致仍然有重复的请求到达服务端,所以还是要在服务端做防重/幂等的处理

    PS:这里额外插入一点我对防重和幂等的理解:防重指的是防止重复提交幂等指的是多次请求如一次,简单说,就是防重可以给对重复请求抛异常,幂等是对重复的请求响应第一次的结果,在我们讨论的这个场景里,幂等 就是响应唯一的订单号。

    在这里插入图片描述

    防重第一步,需要识别请求是否重复,这一步,需要客户端配合实现

    需要客户端在请求下单接口的时候,需要生成一个唯一的请求号:requestId,服务端拿这个请求号,判断是否重复请求

    客户端 怎么生成唯一的请求号?客户端在提交之前,先去 发送请求 获取一个uuid作为requestId,然后请求时带上这个requestId这个参数即可

    利用数据库实现幂等

    可以在订单表t_order里添加一个字段:requestId,添加唯一索引:

    在这里插入图片描述

    利用Redis防重

    另外一个办法,就是下单请求的时候要加锁了,通常我们的服务都是集群部署,所以一般都是用Redis实现分布式锁

    大概的逻辑:

    • 就是以requestId为维度,进行加锁,如果获取锁失败,就抛一个自定义的重复下单异常。
    • 如果获取到锁,先check一下,是否已经下单,为了提高性能,下单完成后,也把下单的结果放在Redis缓存里
      在这里插入图片描述

    大概的代码如下:

        public PlaceOrderResVO placeOrder(PlaceOrderReqVO reqVO) {
            //加锁
            RLock orderLock = redissonClient.getLock(RedisConstant.PLACE_ORDER_LOCK_KEY + reqVO.getRequestId());
            //获取锁失败,抛出重复下单异常
            if(orderLock.isExistes){
              throw new OrderRepeatException();
            }
            // 加锁
            orderLock.lock();
            try {
                //检查是否已经下单
                RBucket<PlaceOrderResVO> orderCache = redissonClient.getBucket(RedisConstant.PLACE_ORDER_LOCK_KEY+reqVO.getRequestId());
                if(orderCache.isExistes){
                    return orderCache.get();
                }
                //下单业务逻辑
                ……
                //落库
                //订单落库
                Order order = orderMapper.saveOrder(orderDO); 
                ……
                //缓存结果
                orderCache.put(resVO);
                return resVO;
            } 
            } catch (Exception e) {
                //……
            } finally {
                orderLock.unlock();
            }
            return resVO;
        }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32

    为什么获取不到锁的时候要抛异常呢?

    因为下单里面其实还有一些其它的业务流程,比如**锁库存、清优惠券……**而此时,获取到锁的请求的下单流程还没有结束,下单的结果还获取不到,没法完成响应,也就没办法做幂等

    客户端,也可以根据响应的状态码,进行特殊处理,比如这个异常先不提示,但是允许用户再次点击下单按钮,来提升用户的体验

  • 相关阅读:
    工作中使用Redis的10种场景
    分布式是什么?
    算力理解MIPS/DMIPS/MFLOPS/TOPS
    Javascript的HTML DOM (文档对象模型)
    【java刷算法】牛客—剑指offer3栈、数组、递归、二分法的初步练习
    【概率论】斗地主中出现炸弹的几率
    深入浅出理解串口
    CV计算机视觉每日开源代码Paper with code速览-2023.10.20
    2310C++仔细区别指针与实例
    如何在矩池云使用 Poetry 管理项目环境
  • 原文地址:https://blog.csdn.net/qq_30659573/article/details/127887868