在订单的创建和支付系统时,会有检查逻辑是否已经签约。如果查到某个客户与某个投资顾问的投顾服务产品已经有签约关系,则不允许重复签约。但是QA反应,因目前客户端可以多终端登陆,QA反应当两个终端同时间提交同一订单时,会对同一产品生成2个订单。
为了避免在同一时间的2个请求生成2个订单,可以通过Redis缓存一个lockkey来生成一个锁。基本思路为:
在开始创建订单前,在redis中缓存一个由客户号clientId+投资顾问adviserId+名下产品productId的 lockey值,创建订单完成后,删除该lockkey值。这样,每次创建订单前先查询 该 “客户号clientId+投资顾问adviserId+名下产品productId” 对应的lockey值在缓存中是否存在,如果存在说明有正在创建中的订单,直接返回。
String lockKey = CacheKeys.genClientAdviserProductConfirmKey(crmAuth.getClientId(), String.valueOf(adviserId), productId);//拼接key值
//检测该订单对应的lockKey值是否存在,存在则返回,不存在则在缓存中缓存lockKey,并开始往下执行订单创建
if(lockService.checkRepeat(lockKey)) {
return JsonResponse.failure("订单创建中,请勿重复提交申请!");
}
//执行订单创建逻辑createOrder()
//订单创建结束后,删除lockKey解锁
lockService.delFromLockList(lockkey);
相应的lockService实现如下:
@Service
@Slf4j
public class LockServiceImpl implements LockService {
@Autowired
private StringRedisTemplate stringRedisTemplate;
private static int expireTime = 15;
@Override
//检查key锁是否存在,不存在则创建key,加锁
public boolean checkRepeat(String cacheKey) {
boolean result = stringRedisTemplate.opsForValue().setIfAbsent(cacheKey, cacheKey);
if (result) {
log.debug("update redis lock key {}", cacheKey);
stringRedisTemplate.expire(cacheKey, expireTime, TimeUnit.SECONDS);
}
return !result;
}
@Override
//删除key 解锁
public void delFromLockList(String cacheKey) {
stringRedisTemplate.delete(cacheKey);
}
}