如何避免重复提交问题

一、简述

所谓幂等性,就是一个接口,多次发起同一个请求,该接口得保证结果是准确的,比如不能多扣款、不能多插入一条数据、不能将统计值多统计 1。这就是幂等性。

1️⃣在编程中常见的幂等
①select 查询天然幂等
②delete 删除也是幂等,删除同一个多次效果一样
③update 直接更新某个值的,幂等
④update 更新累加操作的,非幂等
⑤insert 非幂等操作,每次新增一条

2️⃣产生原因:由于重复点击或者网络重发
①点击提交按钮两次
②点击刷新按钮
③使用浏览器后退按钮重复之前的操作,导致重复提交表单
④使用浏览器历史记录重复提交表单
⑤浏览器重复的HTTP请求
⑥nginx 重发等情况
⑦分布式 RPC 的 try 重发等

二、理解

1️⃣问题背景
分布式系统中的接口,如何保证幂等性?做分布式系统的时候,这是一个必须要考虑的生产环境的技术问题。
假设有个服务提供付款接口,该服务部署在了 5 台机器上。用户在前端上操作的时候,一个订单不小心发起了两次支付请求,该请求分散在了该服务部署的不同的机器上,结果一个订单扣款两次。
或者是订单系统调用支付系统进行支付,结果不小心因为网络超时了,然后订单系统走了重试机制,支付系统收到一个支付请求两次,而且因为负载均衡算法落在了不同的机器上,问题由此而生。

2️⃣问题剖析
这个不是技术问题,这个没有通用的一个方法,这个应该结合业务来保证幂等性。其实保证幂等性主要是三点:

  1. 对于每个请求必须有一个唯一的标识。举个例子:订单支付请求,肯定得包含订单 id,一个订单 id 最多支付一次。
  2. 每次处理完请求之后,必须有一个记录标识这个请求处理过了。常见的方案是在数据库中记录个状态,比如支付之前记录一条这个订单的支付流水。
  3. 每次接收请求需要进行判断,判断之前是否处理过。比如说,如果有一个订单已经支付了,就已经有了一条支付流水,那么如果重复发送这个请求,则此时先插入支付流水,orderId 已经存在了,唯一键约束生效,报错插入不进去的。然后系统就不用再扣款了。

3️⃣实际运作
实际需要结合具体业务来,比如说利用 Redis,用 orderId 作为唯一键。只有成功插入这个支付流水,才可以执行实际的支付扣款。

要求是支付一个订单,必须插入一条支付流水。order_id 建 unique key。用户在支付一个订单之前,先插入一条支付流水,order_id 就已经进去了。系统就可以写一个标识到 Redis 里面去,set order_id payed,下一次重复请求过来了,先查 Redis 的 order_id 对应的 value,如果是 payed 就说明已经支付过了,用户就可以避免重复支付了。

三、解决方案

1️⃣前端 js 提交禁止按钮:可以用一些 js 组件

2️⃣使用Post/Redirect/Get模式
在提交后执行页面重定向,这就是所谓的 Post-Redirect-Get (PRG) 模式。简言之,当用户提交了表单后,去执行一个客户端的重定向,转到提交成功信息页面。这能避免用户按 F5 导致的重复提交,而且也不会出现浏览器表单重复提交的警告,也能消除按浏览器前进和后退导致的同样问题。

3️⃣在 session 中存放一个特殊标志

在服务器端,生成一个唯一的标识符,将它存入 session,同时将它写入表单的隐藏字段中,然后将表单页面发给浏览器,用户录入信息后点击提交,在服务器端,获取表单中隐藏字段的值,与 session 中的唯一标识符比较,相等说明是首次提交,就处理本次请求,然后将 session 中的唯一标识符移除;不相等说明是重复提交,就不再处理。

4️⃣其他借助使用 header 头设置缓存控制头 Cache-control 等方式

比较复杂,不适合移动端 APP 的应用。

5️⃣借助数据库

insert 使用唯一索引。update 使用乐观锁 version 法。这种在大数据量和高并发下效率依赖数据库硬件能力,可针对非核心业务。

6️⃣借助悲观锁

使用select … for update或synchronized锁住先查再 insert 或者 update 一样,但要避免死锁,效率较差。针对单体应用,请求并发不大,可以推荐使用。

四、自定义注解@RreventReSubmit

在传统的 web 项目中,防止重复提交,通常做法是:后端生成一个唯一的提交令牌(uuid),并存储在服务端。页面提交请求携带该提交令牌,后端验证并在第一次验证后删除该令牌,保证提交请求的唯一性。

思路没有问题,但是需要前后端都稍加改动,如果在业务开发完再加这个的话,改动量未免有些大了。无需前端配合,纯后端处理,是最清爽的。设计思路如下:

自定义注解@RreventReSubmit 标记所有Controller中的提交请求。通过AOP对所有标记@RreventReSubmit 的方法拦截。在业务方法执行前,获取当前用户的 token(或者JSessionId)+ 当前请求地址,作为一个唯一 KEY,去获取Redis 分布式锁(如果此时并发获取,只有一个线程会成功获取锁)。当有请求调用接口时,到 Redis 中查找相应的 key。如果能找到,则说明重复提交;如果找不到,则执行操作。业务方法执行后,释放锁。

1️⃣导入aop依赖

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-aop</artifactId>
</dependency>

2️⃣自定义注解

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface PreventReSubmit {
}

3️⃣定义切面类:切面类需要使用 @Aspect 和 @Component 这两个注解做标注。

@Aspect
@Component
@Slf4j
public class UserAspect {@Resourceprivate RedisUtil redisUtil;@Value("${user.session.key}")private String userSessionKey;@Pointcut(value = "@annotation(com.xxp.annotation.RreventReSubmit )")public void annotationPointCut() {}@Around("annotationPointCut()")public Object NoReSubmit(ProceedingJoinPoint joinPoint) {ServletRequestAttributes attributes = 
(ServletRequestAttributes) RequestContextHolder.getRequestAttributes();//获取requestHttpServletRequest request = attributes.getRequest();HttpSession session = request.getSession();//从session中获取登录的user对象,如果为null,则要求重新登录Object sessionUser = session.getAttribute(userSessionKey);if (sessionUser == null) {return Response.FAIL("页面超时,请重新登录");}User user = (User) sessionUser;Integer userId = user.getId();//获取接口的请求参数,如果时Article类型,则保存为Article对象,使用Article对象里的title属性Object[] args = joinPoint.getArgs();Article article = null;for (Object object : args) {if (object instanceof Article) {article = (Article) object;}}if (args == null) {return Response.FAIL("请求参数错误");}//组装redis key 从redis中获取对应的值String key = userId + "_" + article.getTitle();Object flag = redisUtil.getStr(key);//如果redis中不存在对应的值,则执行原有的代码逻辑(插入文章操作)if (flag == null) {//redis设置key,value值为1redisUtil.setStr(key, "1");//设置有效期为5分钟redisUtil.strSetExpireSeconds(key, 5 * 60L);try {return joinPoint.proceed();} catch (Throwable throwable) {redisUtil.delStr(key);return Response.FAIL("系统错误,请联系管理员!");}} else {//如果redis中存在对应的值,则证明重复提交,返回对应的信息log.info("{}:重复提交", key);return Response.FAIL("重复提交");}}
}

在想要防止重复提交的接口上添加注解即可使用。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

生物信息学与智能信息处理2021年学术会议(BIIP2021):会议总结

前言&#xff1a;参加了2021年5月21日至23日于武汉举办的生物信息学与智能信息处理2021年会议&#xff0c;使得我对于生物信息有了更深入的了解。于是&#xff0c;在本文中记录下心得与体会。 一、会议主题 本次会议大主题为生物信息学和智能信息处理&#xff0c;可以细分为三…

均匀”的本质------贝特朗悖论的思考

摘 要 贝特朗悖论是概率论中的著名悖论。文章对古典概型中的无差别原则以及引起争议的贝特朗悖论做出了简要解释和介绍&#xff0c;并通过线性条件修正对悖论的经典计算方法进行驳斥&#xff0c;最后肯定了悖论在数学发展历程中的重要意义。 关键词 贝特朗悖论 无差别原则 概率…

剧情很有趣:安全专家被骗记

导读&#xff1a;这是来自一篇安全宝的文章&#xff0c;我觉得挺有意思的。在现实生活中我就接到过好几次类似的诈骗电话。本文故事性很强&#xff0c;有可读性&#xff0c;发出来也是给大伙提个醒&#xff0c;剧情很有趣。 近 日&#xff0c;国内第三方安全权威公司资深网络工…

云米预计Q3营收同比下降三成,陈小平对未来增长没信心?

8月25日&#xff0c;纳斯达克上市公司云米&#xff08;Nasdaq:VIOT&#xff09;发布了截至6月30日的2021年第二季度财报。财报显示&#xff0c;云米的营收规模开始出现下滑。同时&#xff0c;云米预计2021年第三季度的营收将大幅减少。 贝多财经发现&#xff0c;云米还在财报中…

云米发行区间9到11美元 IPO后陈小平投票权为66%

雷帝网 雷建平 9月12日报道 小米旗下生态链家电供应商云米昨日更新招股书&#xff0c;IPO价格区间为9-11美元&#xff0c;最高募集资金为1.44亿美元。 IPO前&#xff0c;云米创始人陈小平通过Viomi Limited持股为41.3%&#xff0c;加上代持共有50.7%股权&#xff0c;为公司最大…

专访云米CEO陈小平:小米生态链模式的路会越走越宽

雷帝网 雷建平 9月26日报道 小米净水器生产商云米昨日在美国上市&#xff0c;宣告中国“家庭物联网第一股”的诞生。云米上市&#xff0c;也意味着继华米之后&#xff0c;又一家小米生态链企业今年在美国上市。 云米CEO陈小平在上市现场接受雷帝触网创始人雷建平专访时表示&…

云米敲响美国上市钟 陈小平:感谢时代,相信未来

雷帝网 乐天 9月16日报道 小米净水器生产商云米昨日在美国纳斯达克上市&#xff0c;发行价为9美元&#xff0c;以发行价计算&#xff0c;市值超过7亿美元。云米上市也宣告中国“家庭物联网第一股”的诞生。 云米CEO陈小平在上市现场表示&#xff0c;今天是年仅4年的云米在美国纳…

陈小平团队回应疟原虫治癌项目被叫停:仍正常进行

来源&#xff1a;界面 原标题&#xff1a;陈小平团队回应“疟原虫项目被叫停”&#xff1a;仍在正常进行 今年春节至今&#xff0c;备受争议的“疟原虫治疗晚期癌症”事件一直受到各方关注。近日&#xff0c;有消息称&#xff0c;参与该项目临床试验的医院接到中科院通知&…

2023年裁员潮即将来临,当下年轻人该如何自救?官方为此做出回应

自改革开放以来&#xff0c;我国就发生了7次大的裁员潮。 例如&#xff1a;20世纪90年代国企职工裁员潮&#xff1b;2008年金融危机的裁员潮&#xff1b;2021年电子和金融行业裁员潮&#xff1b;2016年O2O裁员潮&#xff1b;2019年互联网行业裁员潮&#xff1b;2020年疫情裁员…

OPPO回应裁员20%传闻:严重不实信息 各方面工作有序开展

新浪科技讯 9月18日下午消息&#xff0c;针对裁员20%的传闻&#xff0c;OPPO向新浪科技回应称&#xff0c;此为严重不实信息&#xff0c;各方面工作有序开展。 近日有传闻称&#xff0c;OPPO已经开始裁员&#xff0c;涉及 IoT业务、互联网业务&#xff0c;IoT、应用商店等非核…

失业率下降!初创公司正大力招揽科技巨头被裁员工

整理 | 朱珂欣 出品 | CSDN程序人生&#xff08;ID&#xff1a;coder_life&#xff09; 当硅谷巨头们“大刀阔斧”开启裁员操作&#xff0c;难免惹人惶恐不安。 1 月 20 日&#xff0c;谷歌母公司 Alphabet 宣布将裁员约 1.2 万人。3 月 14 日&#xff0c;扎克伯格宣布 Met…

层次分析法原理及应用案例

层次分析法是指将一个复杂的多目标决策问题作为一个系统&#xff0c;将目标分解为多个目标或准则&#xff0c;进而分解为多指标&#xff08;或准则、约束&#xff09;的若干层次&#xff0c;通过定性指标模糊量化方法算出层次单排序&#xff08;权数&#xff09;和总排序&#…

【AHP】层次分析法 | 过程解读 案例实践

层次分析法 | 过程解读 案例实践 导读 本文将带领读者了解 AHP 法&#xff0c;通过案例学习&#xff0c;学会使用 AHP 法解决实际问题。在适当的地方将深入了解&#xff0c;例如 AHP 法过程中出现的不一致情况&#xff0c;我们将详细讨论为什么会出现不一致情况&#xff0c;如…

SPSS可信度数据分析

可信度分析 SPSS是常用的统计学数据处理软件&#xff0c;在运用该软件处理数据时会用到数据的可信度分析&#xff0c;通常可信度分析也会在问卷调查等方面运用到&#xff0c;下面是SPSS对于可信度分析的操作 分析-标度-可靠性分析 注意标选统计中的“删除项后的标度”&#x…

职场IT老手教你3步教你玩转可视化大屏设计,让领导眼前一亮!

我是制造企业的IT中心的研发人员&#xff0c;平常工作就是配合业务部门出出报表&#xff0c;选型一些商业软件&#xff0c;并在内部负责实施运维。最近领导出去参观了一些数字化转型比较领先的工厂和制造企业&#xff0c;回来就甩给我几张图&#xff0c;问能不能我们也做几个这…

如何做好软件界面设计?2个方法一次掌握!

软件界面设计通常是指 UI 设计—— User Interface 的简称。UI 设计是指对软件的人机交互、操作逻辑、界面美观的整体设计。 好的软件界面设计要做到清晰、舒适、美观。要想做好软件界面设计&#xff0c;需要遵循软件界面设计的 4 大原则&#xff1a;清晰、高效、一致、明了。同…

想自己设计新房?家庭装修设计和室内布局软件怎么能少!

无论是从事室内设计&#xff0c;还是装修自己的房屋&#xff0c;家庭装修设计和室内布局软件是必须的&#xff0c;下面就为大家介绍几款相关软件&#xff0c;对于想自己设计新房&#xff1f;家庭装修设计和室内布局软件怎么能少&#xff01; Room Arranger for mac(室内布局设计…

ChatGPT搅动AI又一波风潮,却扒出了百度的长板

文|智能相对论 作者| 叶一城 野火燎原&#xff0c;openAI搞出的chatGPT大有当年alphaGo的架势&#xff0c;搅动整个AI业界春心荡漾。 从openAI的大金主微软&#xff0c;到昔日明星alphaGo的爸爸Google&#xff0c;都火急火燎地要出来表态——前者要彰显主权并在内部全面应用&am…

音视频技术开发周刊 | 297

每周一期&#xff0c;纵览音视频技术领域的干货。 新闻投稿&#xff1a;contributelivevideostack.com。 Geenee AR为品牌商和零售商提供虚拟试穿应用 这意味着Geenee AR的虚拟试穿解决方案能够与品牌商现有的销售渠道无缝集成。 谁说苹果掉队了&#xff1f;WWDC上只字未提AI&a…

摩尔定律被提出 | 历史上的今天

整理 | 王启隆 透过「历史上的今天」&#xff0c;从过去看未来&#xff0c;从现在亦可以改变未来。 今天是 2023 年 4 月 19 日&#xff0c;在 1918 年的今天&#xff0c;克利福德贝里&#xff08;Clifford Berry&#xff09;出生。他是世界上第一台电子数字计算设备 Atanasoff…