接口幂等性就是用户对于同一操作发起的一次请求或者多次请求的结果是一致的,不会因为多次点击而产生了副作用;比如说支付场景,用户购买了商品支付扣款成功,但是返回结果的时候网络异常,此时钱已经扣了,用户再次点击按钮,此时会进行第二次扣款,返回结果成功,用户查询余额返发现多扣钱了,流水记录也变成了两条.... 这就没有保证接口的幂等性。
1.用户多次点击按钮;
2.用户页面回退再次提交;
3.微服务互相调用,由于网络问题,导致请求失败。feign触发重试机制;
4.其他业务情况;
不得不提到的危险性:
先删除token还是后删除token;
(1). 先删除可能导致,业务确实没有执行,重试还带上之前token,由于防重设计导致,请求还是不能执行。
(2). 后删除可能导致,业务处理成功,但是服务闪断,出现超时,没有删除token,别人继续重试,导致业务被执行两边
(3). 我们最好设计为先删除token,如果业务调用失败,就重新获取token再次请求。
Token获取、比较和删除必须是原子性
(1)redis.get(token)、token.equals、redis.del(token) 如果这两个操作不是原子,可能导致,高并发下,都get到同样的数据,判断都成功,继续业务并发执行
(2)可以在redis使用lua脚本完成这个操作
if redis.call('get',KEYS[1])==ARGV[1] then return redis.call('del',KEYS[1]) else return 0 end
select * from order where id=1 for update;
悲观锁使用时一般伴随事务一起使用,数据锁定时间可能会很长,需要根据实际情况选用。另外要注意的是,id字段一定是主键或者唯一索引,不然可能造成锁表的结果,处理起来会非常麻烦。
2、数据库乐观锁
这种方法适合在更新的场景中:
update t_goods set count=count-1,version=version+1 where good_id=2 and version=1
根据version版本,也就是在操作库存前先获取当前商品的version版本号,然后操作的时候带上此version号。我们梳理下,我们第一次操作库存时,得到version为1,调用库存服务version变成了2;但返回给订单服务出现了问题,订单服务又一次发起调用库存服务,当订单服务传如的version还是1,再执行上面的sql语句时,就不会执行;因为version已经变为2了,where条件就不成立。这样就保证了不管调用几次,只会真正的处理一次。
注:乐观锁主要使用于处理读多写少的问题。
3、业务层分布式锁
如果多个机器可能在同一时间同时处理相同的数据,比如多台机器定时任务都拿到了相同数据处理,我们就可以加分布式锁,锁定此数据,处理完成后释放锁。获取到锁的必须先判断这个数据是否被处理过。
1、数据库唯一约束
插入数据,应该按照唯一索引进行插入,比如订单号,相同的订单就不可能有两条记录插入。我们在数据库层面防止重复。这个机制是利用了数据库的主键唯一约束的特性,解决了在insert场景时幂等问题。但主键的要求不是自增的主键,这样就需要业务生成全局唯一的主键。如果是分库分表场景下,路由规则要保证相同请求下,落地在同一个数据库和同一表中,要不然数据库主键约束就不起效果了,因为是不同的数据库和表主键不相关。
2、redis set 防重
很多数据需要处理,只能被处理一次,比如我们可以计算数据的MD5将其放入redis的set,每次处理数据,先看这个MD5是否已经存在,存在就不处理。
之前说的redis防重也算
# nginx 中的配置
proxy_set_header X-Request-Id $request_id;
介绍一下业务场景,我们从购物车页面点击结算开始到订单页面,然后在订单页面点击“提交订单”按钮后来生成订单。我们就以这个实际项目中的流程为例子,用伪代码的形式来说明。
我们选用token的方式来做验证订单重复提交的幂等性问题。
//你可以把这个前缀做到一个常量类里面去
String USER_ORDER_TOKEN_PREFIX = "order:token";
//获取当前用户登录的信息
MemberResponseVo memberResponseVo = LoginUserInterceptor.loginUser.get();
//TODO 5、防重令牌(防止表单重复提交)
//为用户设置一个token,三十分钟过期时间(存在redis)
String token = UUID.randomUUID().toString().replace("-", "");
// memberResponseVo.getId() 为用户ID
redisTemplate.opsForValue().set(USER_ORDER_TOKEN_PREFIX+memberResponseVo.getId(),token,30, TimeUnit.MINUTES);
//1、验证令牌是否合法【令牌的对比和删除必须保证原子性】
//这就是lua脚本
String script = "if redis.call('get', KEYS[1]) == ARGV[1] then return redis.call('del', KEYS[1]) else return 0 end";
//获取我们前台下单时传过来的tooken
String orderToken = vo.getOrderToken();
//通过lure脚本原子验证令牌和删除令牌。并在最终返回 0 或 1
Long result = redisTemplate.execute(
new DefaultRedisScript<Long>(script, Long.class),//传递一个lua脚本+返回值类型
Arrays.asList(USER_ORDER_TOKEN_PREFIX + memberResponseVo.getId()),//因为我们可以看到,lua脚本里面是用的KEYS[1] 来接收的,所以我们需要传入一个Array类型,然后lua脚本会去取第一个值作为key去查询
orderToken//这是我们要对照的值(前台下单时传过来的tooken)
);
//验证
if (result == 0L) {
//令牌验证失败
responseVo.setCode(1);
return responseVo;
} else {
//这里继续执行后续的业务,省略...........
}
这样,经过tooken验证之后我们就完美的解决了幂等问题,并且不会对数据库服务器造成压力。