重复点击提交、产生多笔数据、保持数据只操作一次---->接口幂等性校验

简介: 重复点击提交、产生多笔数据、保持数据只操作一次---->接口幂等性校验

一、工作真实场景 + 常出现场景

  1. 真实场景:在一次工作中进行成品出库创建成品出库单时,手抖了一下,重复点击了两次确定(提交表单)。结果很神奇的发现居然产生了两笔一模一样的数据(流水号都一样),当时就很懵逼,稍作思考,想想应该是在同一时刻创建了两个出库单。感觉很有意思(因为之前没有遇到过,写代码的时候也没有考虑到这个问题的发生),后面换了一种方式复现场景:提交表单时多次点击Enter按钮,还是会产生多笔重复数据。
  2. 常见场景:
    订单接口, 不能多次创建订单
    支付接口, 重复支付同一笔订单只能扣一次钱
    支付宝回调接口, 可能会多次回调, 必须处理重复回调
    普通表单提交接口, 因为网络超时等原因多次点击提交, 只能成功一次
    等等

二、工作解决方案 + 百度解决方案

  1. 工作:
    (1) 在对应业务接口上加上@Transactional,后面发现不能要(只是为了在遇到错误时之前对数据库操作做一个回滚),因为在做相同数据校验的时候去数据库是查不到同比数据的,此时所有代码没走完,事务未提交。所以只能拿掉事务。
    (2)在对应的业务代码块加上synchronized (this) {}对业务代码进行代码块的同步锁,锁里面做唯一性校验(查询数据库是否有相同的流水号[流水号是按照英文前缀加上时间戳实现],有的话直接抛异常)。后面发现如果在极端条件下这个判断还是有一定缺陷(时间戳只是精确到秒)。
@Override
    public JsonData saveOrUpdate(OutboundProductReq outboundProductReq) {
        // 出库单不能为空
        if (StringUtils.isBlank(outboundProductReq.getOutboundName())) {
            return JsonData.buildResult(BizCodeEnum.STORE_PRODUCT_OUTBOUND_NAME_EMPTY);
        }
        OutboundProductDO outboundProductDO;
        if (outboundProductReq.getId() != null && outboundProductReq.getId() > 0) {
            outboundProductDO = outboundProductMapper.selectById(outboundProductReq.getId());
            Integer outboundCount = outboundProductDO.getOutboundCount();
            // 校验成品库存
            JsonData data = checkProductStock(outboundProductReq.getProductId(), outboundProductReq.getOutboundCount(), outboundCount);
            if (data.getCode() == 0) {
                // 修改
                if (outboundProductDO.getOutboundState() == 0) {
                    BeanUtils.copyProperties(outboundProductReq, outboundProductDO);
                    outboundProductMapper.updateById(outboundProductDO);
                    // 修改库存详情表
                    updateTODetailProduct(outboundProductDO);
                    // 修改成品冻结数
                    StockProductDO stockProductDO = stockProductMapper.selectOne(new QueryWrapper<StockProductDO>().eq("product_id", outboundProductDO.getProductId()));
                    stockProductDO.setFreezeCount(stockProductDO.getFreezeCount() - outboundCount + outboundProductReq.getOutboundCount());
                    stockProductMapper.updateById(stockProductDO);
                } else {
                    return JsonData.buildResult(BizCodeEnum.STORE_PRODUCT_NOT_MODIFY);
                }
            } else {
                return data;
            }
        } else {
            // 校验成品库存
            JsonData data = checkProductStock(outboundProductReq.getProductId(), outboundProductReq.getOutboundCount());
            if (data.getCode() == 0) {
                // 新增
                synchronized (this) {
                    // 问题:连续点击速度很快时,会出现两条重复记录 解决方案:将以下操作抽取出来做方法,在方法上加事务和锁。
                    if (outboundProductMapper.selectOne(new QueryWrapper<OutboundProductDO>()
                            .eq("serial_code", CommonUtil.getCurrentSerialNumber("SN_OUT"))) != null) {
                        return JsonData.buildResult(BizCodeEnum.STORE_PRODUCT_OUTBOUND_SAME);
                    }
                    outboundProductDO = new OutboundProductDO();
                    BeanUtils.copyProperties(outboundProductReq, outboundProductDO);
                    outboundProductDO.setOutboundId(UUID.randomUUID().toString());
                    outboundProductDO.setOutboundState(0);
                    outboundProductDO.setOutboundType(0);
                    outboundProductDO.setSerialCode(CommonUtil.getCurrentSerialNumber("SN_OUT"));
                    outboundProductMapper.insert(outboundProductDO);
                    // 冻结成品库存
                    StockProductDO stockProductDO = stockProductMapper.selectOne(new QueryWrapper<StockProductDO>()
                            .eq("product_id", outboundProductDO.getProductId()));
                    stockProductDO.setFreezeCount(outboundProductReq.getOutboundCount() + stockProductDO.getFreezeCount());
                    stockProductMapper.updateById(stockProductDO);
                    // 保存到库存详情表
                    saveTODetailProduct(outboundProductDO, OpTypeEnum.OP_LOCK);
                }
            } else {
                return data;
            }
        }
        return JsonData.buildSuccess();
    }
  1. 唯一索引 – 防止新增脏数据
    token机制 – 防止页面重复提交
    悲观锁 – 获取数据的时候加锁(锁表或锁行)
    乐观锁 – 基于版本号version实现, 在更新数据那一刻校验数据
    分布式锁 – redis(jedis、redisson)或zookeeper实现
    状态机 – 状态变更, 更新数据时判断状态

分享优秀博客

幂等校验

相关文章
|
NoSQL 关系型数据库 MySQL
接口防刷 && 接口幂等性问题
接口防刷 && 接口幂等性问题
284 0
|
Java API Maven
敏感数据的保护伞——SpringBoot Jasypt加密库的使用
我们经常会在yml配置文件中存放一些敏感数据,比如数据库的用户名、密码,第三方应用的秘钥等等。这些信息直接以明文形式展示在文件中,无疑是存在较大的安全隐患的,所以今天这篇文章,我会借助jasypt实现yml文件中敏感信息的加密处理。
5511 1
敏感数据的保护伞——SpringBoot Jasypt加密库的使用
|
7月前
|
人工智能 关系型数据库 MySQL
轻松搭建AI知识问答系统,阿里云PolarDB MCP深度实践
无论是PolarDB MySQL兼容MySQL语法的SQL执行功能,还是其特有的OLAP分析与AI能力,通过MCP协议向LLM开放接口后,显著降低了用户使用门槛,更为未来基于DB-Agent的智能体开发奠定了技术基础
|
人工智能 Java Serverless
【MCP教程系列】搭建基于 Spring AI 的 SSE 模式 MCP 服务并自定义部署至阿里云百炼
本文详细介绍了如何基于Spring AI搭建支持SSE模式的MCP服务,并成功集成至阿里云百炼大模型平台。通过四个步骤实现从零到Agent的构建,包括项目创建、工具开发、服务测试与部署。文章还提供了具体代码示例和操作截图,帮助读者快速上手。最终,将自定义SSE MCP服务集成到百炼平台,完成智能体应用的创建与测试。适合希望了解SSE实时交互及大模型集成的开发者参考。
13802 60
|
8月前
|
人工智能 Java 数据库
如何保证接口幂等性?
在分布式系统中,接口幂等性至关重要。本文详解其定义、重要性及实现方案,包括唯一索引、Token机制、分布式锁、状态机与版本号机制,并提供最佳实践建议,助你提升系统可靠性与用户体验。
1512 1
|
监控 Java
压力测试Jmeter的简单使用,性能监控-堆内存与垃圾回收 -jvisualvm的使用
这篇文章介绍了如何使用JMeter进行压力测试,包括测试前的配置、测试执行和结果查看。同时,还探讨了性能监控工具jconsole和jvisualvm的使用,特别是jvisualvm,它可以监控内存泄露、跟踪垃圾回收、执行时内存和CPU分析以及线程分析等,文章还提供了使用这些工具的详细步骤和说明。
压力测试Jmeter的简单使用,性能监控-堆内存与垃圾回收 -jvisualvm的使用
|
设计模式 缓存 前端开发
什么是幂等性?四种接口幂等性方案详解!
本文深入分布式系统中的幂等性问题及其解决方案,涵盖数据库唯一主键、乐观锁、PRG模式和防重Token等方法,关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
什么是幂等性?四种接口幂等性方案详解!
|
XML 前端开发 Java
Spring,SpringBoot和SpringMVC的关系以及区别 —— 超准确,可当面试题!!!也可供零基础学习
本文阐述了Spring、Spring Boot和Spring MVC的关系与区别,指出Spring是一个轻量级、一站式、模块化的应用程序开发框架,Spring MVC是Spring的一个子框架,专注于Web应用和网络接口开发,而Spring Boot则是对Spring的封装,用于简化Spring应用的开发。
3967 0
Spring,SpringBoot和SpringMVC的关系以及区别 —— 超准确,可当面试题!!!也可供零基础学习
|
缓存 监控 负载均衡
将近2万字的Dubbo原理解析,彻底搞懂dubbo
市面上有很多基于RPC思想实现的框架,比如有Dubbo。今天就从Dubbo的SPI机制、服务注册与发现源码及网络通信过程去深入剖析下Dubbo。
29702 9