高并发下保证接口幂等性的策略

前言

1.接口幂等性问题常见场景:

  1. 有时我们在填写某些form表单时,保存按钮不小心快速点了两次,表中竟然产生了两条重复的数据,只是id不一样。
  2. 我们在项目中为了解决接口超时问题,通常会引入了重试机制。第一次请求接口超时了,请求方没能及时获取返回结果,为了避免返回错误的结果,于是会对该请求重试几次,这样也会产生重复的数据。
  3. mq消费者在读取消息时,有时候会读取到重复消息,如果处理不好,也会产生重复的数据。

没错,这些都是幂等性问题。

接口幂等性是指用户对于同一操作发起的一次请求或者多次请求的结果是一致的,不会因为多次点击而产生了副作用。

这类问题多发于接口的:

  • insert操作,这种情况下多次请求,可能会产生重复数据。
  • update操作,如果只是单纯的更新数据,比如:update user set status=1 where id=1,是没有问题的。如果还有计算,比如:update user set status=status+1 where id=1,这种情况下多次请求,可能会导致数据错误。

2.接口幂等性和防重设计的区别

防重设计幂等设计,其实是有区别的。防重设计主要为了避免产生重复数据,对接口返回没有太多要求。而幂等设计除了避免产生重复数据之外,还要求每次请求都返回一样的结果。

1. insert前先select(不适用于并发场景)

通常情况下,在保存数据的接口中,我们为了防止产生重复数据,一般会在insert前,先根据namecode字段select一下数据。如果该数据已存在,则不执行update操作,如果不存在,才执行 insert操作。

该方案可能是我们平时在防止产生重复数据时,使用最多的方案。但是该方案不适用于并发场景,在并发场景中,要配合其他方案一起使用,否则同样会产生重复数据。我在这里提一下,是为了避免大家踩坑。

2. 加悲观锁(只防重)

在支付场景中,用户A的账号余额有150元,想转出100元,正常情况下用户A的余额只剩50元。一般情况下,sql是这样的:

update user amount = amount-100 where id=123;

如果出现多次相同的请求,可能会导致用户A的余额变成负数。这种情况,用户A来可能要哭了。于此同时,系统开发人员可能也要哭了,因为这是很严重的系统bug。

为了解决这个问题,可以加悲观锁,将用户A的那行数据锁住,在同一时刻只允许一个请求获得锁,更新数据,其他的请求则等待。

通常情况下通过如下sql锁住单行数据:

select * from user id=123 for update;

具体流程如下:

具体步骤:

  1. 多个请求同时根据id查询用户信息。
  2. 判断余额是否不足100,如果余额不足,则直接返回余额不足。
  3. 如果余额充足,则通过for update再次查询用户信息,并且尝试获取锁。
  4. 只有第一个请求能获取到行锁,其余没有获取锁的请求,则等待下一次获取锁的机会。
  5. 第一个请求获取到锁之后,判断余额是否不足100,如果余额足够,则进行update操作。
  6. 如果余额不足,说明是重复请求,则直接返回成功。

需要特别注意的是:如果使用的是mysql数据库,存储引擎必须用innodb,因为它才支持事务。此外,这里id字段一定要是主键或者唯一索引,不然会锁住整张表。

悲观锁需要在同一个事务操作过程中锁住一行数据,如果事务耗时比较长,会造成大量的请求等待,影响接口性能

此外,每次请求接口很难保证都有相同的返回值,所以不适合幂等性设计场景,但是在防重场景中是可以的使用的。

  • 使用悲观锁可以防止重复数据的产生,但它不能保证每次请求都会返回相同的结果。例如,如果一个操作涉及多个步骤,其中某些步骤可能依赖于外部条件或状态变化,那么即使加了悲观锁,每次请求的结果也可能不同。

只防重不幂等性例子说明:

  • 假设有一个转账操作,从账户A向账户B转账100元。

3. 加乐观锁

既然悲观锁有性能问题,为了提升接口性能,我们可以使用乐观锁。需要在表中增加一个timestamp或者version字段,这里以version字段为例。

在更新数据之前先查询一下数据:

select id,amount,version from user id=123;

如果数据存在,假设查到的version等于1,再使用idversion字段作为查询条件更新数据:

update user set amount=amount+100,version=version+1where id=123 and version=1;

更新数据的同时vers

剩余60%内容,订阅专栏后可继续查看/也可单篇购买

本专栏价格永远为39.9元! 不想当架构师的后端开发工程师不是好码农! 此专栏一方面用于存放我的架构设计学习笔记, 另外我会在本专栏加入一系列最常问八股问题帖子,内容就是我根据自己的面试经历和网上的面经,去筛选八股里面哪些是最常被问到的问题把它们整理出来,大家可以在面试前一两个小时快速把这一系列最常问八股的帖子拿出来看看,临时抱佛脚的效果应该很好

全部评论
aop+redis锁怎么样呢
点赞 回复 分享
发布于 2024-07-26 11:10 四川

相关推荐

船长想实习:我啥技术不会决定去试试,然后进去也不干活就搅局可以吗?
点赞 评论 收藏
分享
评论
4
31
分享

创作者周榜

更多
正在热议
更多
# AI面会问哪些问题? #
24674次浏览 486人参与
# 中国电信笔试 #
31049次浏览 283人参与
# 开放七大实习专项,百度暑期实习值得冲吗 #
14099次浏览 209人参与
# 你的实习产出是真实的还是包装的? #
18722次浏览 329人参与
# 如果秋招能重来,我会____ #
96681次浏览 500人参与
# 春招至今,你的战绩如何? #
59649次浏览 537人参与
# 米连集团26产品管培生项目 #
12935次浏览 285人参与
# i人适合做什么工作 #
36898次浏览 124人参与
# 我是面试官,请用一句话让我破防 #
79502次浏览 219人参与
# 哪些公司真双非友好? #
69189次浏览 287人参与
# 找AI工作可以去哪些公司? #
7643次浏览 183人参与
# 从事AI岗需要掌握哪些技术栈? #
7615次浏览 244人参与
# 面试尴尬现场 #
220747次浏览 861人参与
# 投递几十家公司,到现在0offer,大家都一样吗 #
339873次浏览 2165人参与
# 五一之后,实习真的很难找吗? #
102794次浏览 584人参与
# 金三银四,你的春招进行到哪个阶段了? #
21531次浏览 277人参与
# 你做过最难的笔试是哪家公司 #
29928次浏览 186人参与
# 你小时候最想从事什么职业 #
159835次浏览 2072人参与
# 阿里笔试 #
176285次浏览 1302人参与
# 应届生第一份工资要多少合适 #
20473次浏览 84人参与
# 一张图晒出你司的标语 #
3793次浏览 71人参与
# 面试被问期望薪资时该如何回答 #
382455次浏览 2163人参与
牛客网
牛客网在线编程
牛客网题解
牛客企业服务