高并发下的幂等性问题
1.什么是幂等性
幂等(idempotent、idempotence)是一个数学与计算机学概念,常见于抽象代数中。在编程中.一个幂等操作的特点是其任意多次执行所产生的影响均与一次执行的影响相同。幂等函数,或幂等方法,是指可以使用相同参数重复执行,并能获得相同结果的函数。这些函数不会影响系统状态,也不用担心重复执行会对系统造成改变。例如,“getUsername()和setTrue()”函数就是一个幂等函数. 更复杂的操作幂等保证是利用唯一交易号(流水号)实现. 我的理解:幂等就是一个操作,不论执行多少次,产生的效果和返回的结果都是一样的 。
而在我们实际开发中用到最多的是接口这一块,随着前后端分离的趋势化,接口开发是不可避免的,这里面就涉及到幂等性问题比如:
(1)订单接口, 不能多次创建订单
(2)支付接口, 重复支付同一笔订单只能扣一次钱
(3)支付宝回调接口, 可能会多次回调, 必须处理重复回调
(4)普通表单提交接口, 因为网络超时等原因多次点击提交, 只能成功一次
常用的解决方法:
(1)唯一索引 – 防止新增脏数据
(2)token机制 – 防止页面重复提交
(3)悲观锁 – 获取数据的时候加锁(锁表或锁行)
(4)乐观锁 – 基于版本号version实现, 在更新数据那一刻校验数据
(5)分布式锁 – redis(jedis、redisson)或zookeeper实现
(6)状态机 – 状态变更, 更新数据时判断状态
悲观锁和乐观锁可以看下面链接:
https://blog.csdn.net/qq_34136709/article/details/107911749
我们这里先学习token机制,即通过redis + token机制实现接口幂等性校验
2.实现思路
为需要保证幂等性的每一次请求创建一个唯一标识token, 先获取token, 并将此token存入redis, 请求接口时, 将此token放到header或者作为请求参数请求接口, 后端接口判断redis中是否存在此token:
(1)如果存在, 正常处理业务逻辑, 并从redis中删除此token, 那么, 如果是重复请求, 由于token已被删除, 则不能通过校验, 返回请勿重复操作提示
(2)如果不存在, 说明参数不合法或者是重复请求, 返回提示即可
3.代码如下:
pom文件:
<dependency>
<groupId>redis.clients</groupId>
<artifactId>jedis</artifactId>
<version>2.9.0</version>
</dependency>
JedisUtil:
package com.cp.springbootone.util;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPool;
/**
* @author ChengPeng
* @create 2020-07-31 11:19
*/
@Component
@Slf4j
public class JedisUtil {
@Autowired(required = false)
private JedisPool jedisPool;
private Jedis getJedis(){
return jedisPool.getResource();
}
/**
* 设值
*
* @param key
* @param value
* @return
*/
public String set(String key, String value) {
Jedis jedis = null;
try {
jedis = getJedis();
return jedis.set(key, value);
} catch (Exception e) {
log.error("set key:{} value:{} error", key, value, e);
return null;
} finally {
close(jedis);
}
}
/**
* 设值
*
* @param key
* @param value
* @param expireTime 过期时间, 单位: s
* @return
*/
public String set(String key, String value, int expireTime) {
Jedis jedis = null;
try {
jedis = getJedis();
return jedis.setex(key, expireTime, value);
} catch (Exception e) {
log.error("set key:{} value:{} expireTime:{} error", key, value, expireTime, e);
return null;
} finally {
close(jedis);
}
}
/**
* 设值
*
* @param key
* @param value
* @return
*/
public Long setnx(String key, String value) {
Jedis jedis = null;
try {
jedis = getJedis();
return jedis.setnx(key, value);
} catch (Exception e) {
log.error("set key:{} value:{} error", key, value, e);
return null;
} finally {
close(jedis);
}
}
/**
* 取值
*
* @param key
* @return
*/
public String get