后端面试每日一题 保证幂等性

  • 前端拦截

  • 使用数据库实现幂等性

  • 使用 JVM 锁实现幂等性

  • 使用分布式锁实现幂等性

前端拦截

前端拦截是指通过 Web 站点的页面进行请求拦截,比如在用户点击完“提交”按钮后,我们可以把按钮设置为不可用或者隐藏状态,避免用户重复点击。

执行效果如下图所示:

img

按钮点击效果图

核心的实现代码如下:

但前端拦截有一个致命的问题,如果是懂行的程序员或者黑客可以直接绕过页面的 JS 执行,直接模拟请求后端的接口,这样的话,我们前端的这些拦截就不能生效了。因此除了前端拦截一部分正常的误操作之外,后端的验证必不可少。

数据库实现

数据库实现幂等性的方案有三个:

  • 通过悲观锁来实现幂等性

  • 通过唯一索引来实现幂等性

  • 通过乐观锁来实现幂等性

JVM 锁实现

JVM 锁实现是指通过 JVM 提供的内置锁如 Lock 或者是 synchronized 来实现幂等性。使用 JVM 锁来实现幂等性的一般流程为:首先通过 Lock 对代码段进行加锁操作,然后再判断此订单是否已经被处理过,如果未处理则开启事务执行订单处理,处理完成之后提交事务并释放锁,执行流程如下图所示:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-LbETItUp-1622094717675)(https://img.kancloud.cn/30/31/3031eeabd5f326b1847383d900604923_2254x328.png)]

JVM 锁执行流程图

JVM 锁存在的最大问题在于,它只能应用于单机环境,因为 Lock 本身为单机锁,所以它就不适应于分布式多机环境。

分布式锁实现

分布式锁实现幂等性的逻辑是,在每次执行方法之前先判断是否可以获取到分布式锁,如果可以,则表示为第一次执行方法,否则直接舍弃请求即可,执行流程如下图所示:

img

分布式锁执行流程图

需要注意的是分布式锁的 key 必须为业务的唯一标识,我们通常使用 Redis 或者 ZooKeeper 来实现分布式锁;如果使用 Redis 的话,则用 set 命令来创建和获取分布式锁,执行示例如下:

127.0.0.1:6379> set lock true ex 30 nx

OK # 创建锁成功

其中,ex 是用来设置超时时间的;而 nx 是 not exists 的意思,用来判断键是否存在。如果返回的结果为“OK”,则表
示创建锁成功,否则表示重复请求,应该舍弃

    1. 幂等性注意事项

幂等性的实现与判断需要消耗一定的资源,因此不应该给每个接口都增加幂等性判断,要根据实际的业务情况和操作类型来进行区分。例如,我们在进行查询操作和删除操作时就无须进行幂等性判断。查询操作查一次和查多次的结果都是一致的,因此我们无须进行幂等性判断。删除操作也是一样,删除一次和删除多次都是把相关的数据进行删除(这里的删除指的是条件删除而不是删除所有数据),因此也无须进行幂等性判断。

    1. 幂等性的关键步骤

实现幂等性的关键步骤分为以下三个:

  1. 每个请求操作必须有唯一的 ID,而这个 ID 就是用来表示此业务是否被执行过的关键凭证,例如,订单支付业务的请求,就要使用订单的 ID 作为幂等性验证的 Key;

  2. 每次执行业务之前必须要先判断此业务是否已经被处理过;

  3. 第一次业务处理完成之后,要把此业务处理的状态进行保存,比如存储到 Redis 中或者是数据库中,这样才能防止业务被重复处理。

    1. 数据库实现幂等性

使用数据库实现幂等性的方法有三种:

  • 通过悲观锁来实现幂等性

  • 通过唯一索引来实现幂等性

  • 通过乐观锁来实现幂等性

接下来我们分别来看这些实现方式的具体执行过程。

  • ① 悲观锁

使用悲观锁实现幂等性,一般是配合事务一起来实现,在没有使用悲观锁时,我们通常的执行过程是这样的,首先来判断数据的状态,执行 SQL 如下:

select status from table_name where id=‘xxx’;

然后再进行添加操作:

insert into table_name (id) values (‘xxx’);

最后再进行状态的修改:

update table_name set status=‘xxx’;

但这种情况因为是非原子操作,所以在高并发环境下可能会造成一个业务被执行两次的问题,当一个程序在执行中时,而另一个程序也开始状态判断的操作。因为第一个程序还未来得及更改状态,所以第二个程序也能执行成功,这就导致一个业务被执行了两次。

在这种情况下我们就可以使用悲观锁来避免问题的产生,实现 SQL 如下所示:

begin; # 1.开始事务

select * from table_name where id=‘xxx’ for update; # 2.查询状态

insert into table_name (id) values (‘xxx’); # 3.添加操作

update table_name set status=‘xxx’; # 4.更改操作

commit; # 5.提交事务

  • 10
    点赞
  • 19
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值