唯一性
redis本身天然是单线程的
package com.social.member.server.common;
import com.alibaba.nacos.client.utils.StringUtils;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Component;
@Component
@Slf4j
public class RedisUtil {
@Autowired
private RedisTemplate redisTemplate;
public boolean getlock(String key, String value) {
//如果key值不存在,则返回 true,且设置 value
if (redisTemplate.opsForValue().setIfAbsent(key, value)) {
return true;
}
//获取key的值,判断是是否超时
String curVal = (String) redisTemplate.opsForValue().get(key);
if (StringUtils.isNotEmpty(curVal) && Long.parseLong(curVal) < System.currentTimeMillis()) {
//获得之前的key值,同时设置当前的传入的value。这个地方可能几个线程同时过来,但是redis本身天然是单线程的,所以getAndSet方法还是会安全执行,
//首先执行的线程,此时curVal当然和oldVal值相等,因为就是同一个值,之后该线程set了自己的value,后面的线程就取不到锁了
String oldVal = (String) redisTemplate.opsForValue().getAndSet(key, value);
if(StringUtils.isNotEmpty(oldVal) && oldVal.equals(curVal)) {
return true;
}
}
return false;
}
/**
* 解锁
* @param key
* @param value
*/
public void unlock(String key, String value) {
try {
String curVal = (String) redisTemplate.opsForValue().get(key);
if (StringUtils.isNotEmpty(curVal) && curVal.equals(value)) {
redisTemplate.opsForValue().getOperations().delete(key);
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
生成唯一编码
public BaseResultResponse getCouponCode(@RequestParam String param) {
log.info("CouponManageController getCouponCodebegin");
log.info("CouponManageController getCouponCode入参param;"