据说微信搜寻《Java鱼仔》会变更强哦!
本文收录于JavaStarter ,外面有我残缺的Java系列文章,学习或面试都能够看看哦
每天一个知识点
什么是接口的幂等性,如何实现接口幂等性?
(一)幂等性概念
幂等性本来是数学上的概念,用在接口上就能够了解为:同一个接口,屡次收回同一个申请,必须保障操作只执行一次。
调用接口产生异样并且反复尝试时,总是会造成零碎所无奈接受的损失,所以必须阻止这种景象的产生。
比方上面这些状况,如果没有实现接口幂等性会有很重大的结果:
领取接口,反复领取会导致屡次扣钱
订单接口,同一个订单可能会屡次创立。
(二)幂等性的解决方案
惟一索引
应用惟一索引能够防止脏数据的增加,当插入反复数据时数据库会抛异样,保障了数据的唯一性。
乐观锁
这里的乐观锁指的是用乐观锁的原理去实现,为数据字段减少一个version字段,当数据须要更新时,先去数据库里获取此时的version版本号
select version from tablename where xxx
更新数据时首先和版本号作比照,如果不相等阐明曾经有其余的申请去更新数据了,提醒更新失败。
update tablename set count=count+1,version=version+1 where version=#{version}
乐观锁
乐观锁能够实现的往往用乐观锁也能实现,在获取数据时进行加锁,当同时有多个反复申请时其余申请都无奈进行操作
分布式锁
幂等的实质是分布式锁的问题,分布式锁失常能够通过redis或zookeeper实现;在分布式环境下,锁定全局惟一资源,使申请串行化,理论体现为互斥锁,避免反复,解决幂等。
token机制
token机制的核心思想是为每一次操作生成一个唯一性的凭证,也就是token。一个token在操作的每一个阶段只有一次执行权,一旦执行胜利则保留执行后果。对反复的申请,返回同一个后果。token机制的利用非常宽泛。
(三)token机制的实现
这里展现通过token机制实现接口幂等性的案例:github文末自取
首先引入须要的依赖:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>
<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-lang3</artifactId>
<version>3.4</version>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
3.1、配置申请的办法体和枚举类
首先配置一下通用的申请返回体
public class Response {
private int status;
private String msg;
private Object data;
//省略get、set、toString、无参有参构造方法
}
以及返回code
public enum ResponseCode {
// 通用模块 1xxxx
ILLEGAL_ARGUMENT(10000, "参数不非法"),
REPETITIVE_OPERATION(10001, "请勿反复操作"),
;
ResponseCode(Integer code, String msg) {
this.code = code;
this.msg = msg;
}
private Integer code;
private String msg;
public Integer getCode() {
return code;
}
public void setCode(Integer code) {
this.code = code;
}
public String getMsg() {
return msg;
}
public void setMsg(String msg) {
this.msg = msg;
}
}
3.2 自定义异样以及配置全局异样类
public class ServiceException extends RuntimeException{
private String code;
private String msg;
//省略get、set、toString以及构造方法
}
配置全局异样捕捉器
@ControllerAdvice
public class MyControllerAdvice {
@ResponseBody
@ExceptionHandler(ServiceException.class)
public Response serviceExceptionHandler(ServiceException exception){
Response response=new Response(Integer.valueOf(exception.getCode()),exception.getMsg(),null);
return response;
}
}
3.3 编写创立Token和验证Token的接口以及实现类
@Service
public interface TokenService {
public Response createToken();
public Response checkToken(HttpServletRequest request);
}
具体实现类,外围的业务逻辑都写在正文中了
@Service
public class TokenServiceImpl implements TokenService {
@Autowired
private RedisTemplate redisTemplate;
@Override
public Response createToken() {
//生成uuid当作token
String token = UUID.randomUUID().toString().replaceAll("-","");
//将生成的token存入redis中
redisTemplate.opsForValue().set(token,token);
//返回正确的后果信息
Response response=new Response(0,token.toString(),null);
return response;
}
@Override
public Response checkToken(HttpServletRequest request) {
//从申请头中获取token
String token=request.getHeader("token");
if (StringUtils.isBlank(token)){
//如果申请头token为空就从参数中获取
token=request.getParameter("token");
//如果都为空抛出参数异样的谬误
if (StringUtils.isBlank(token)){
throw new ServiceException(ResponseCode.ILLEGAL_ARGUMENT.getCode().toString(),ResponseCode.ILLEGAL_ARGUMENT.getMsg());
}
}
//如果redis中不蕴含该token,阐明token曾经被删除了,抛出申请反复异样
if (!redisTemplate.hasKey(token)){
throw new ServiceException(ResponseCode.REPETITIVE_OPERATION.getCode().toString(),ResponseCode.REPETITIVE_OPERATION.getMsg());
}
//删除token
Boolean del=redisTemplate.delete(token);
//如果删除不胜利(曾经被其余申请删除),抛出申请反复异样
if (!del){
throw new ServiceException(ResponseCode.REPETITIVE_OPERATION.getCode().toString(),ResponseCode.REPETITIVE_OPERATION.getMsg());
}
return new Response(0,"校验胜利",null);
}
}
3.4 配置自定义注解
这是比拟重要的一步,通过自定义注解在须要实现接口幂等性的办法上增加此注解,实现token验证
@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface ApiIdempotent {
}
接口拦截器
public class ApiIdempotentInterceptor implements HandlerInterceptor {
@Autowired
private TokenService tokenService;
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
if (!(handler instanceof HandlerMethod)) {
return true;
}
HandlerMethod handlerMethod= (HandlerMethod) handler;
Method method=handlerMethod.getMethod();
ApiIdempotent methodAnnotation=method.getAnnotation(ApiIdempotent.class);
if (methodAnnotation != null){
// 校验通过放行,校验不通过全局异样捕捉后输入返回后果
tokenService.checkToken(request);
}
return true;
}
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
}
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
}
}
3.5 配置拦截器以及redis
配置webConfig,增加拦截器
@Configuration
public class WebConfig implements WebMvcConfigurer {
@Override
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(apiIdempotentInterceptor());
}
@Bean
public ApiIdempotentInterceptor apiIdempotentInterceptor() {
return new ApiIdempotentInterceptor();
}
}
配置redis,使得中文能够失常传输
@Configuration
public class RedisConfig {
//自定义的redistemplate
@Bean(name = "redisTemplate")
public RedisTemplate<String,Object> redisTemplate(RedisConnectionFactory factory){
//创立一个RedisTemplate对象,为了不便返回key为string,value为Object
RedisTemplate<String,Object> template = new RedisTemplate<>();
template.setConnectionFactory(factory);
//设置json序列化配置
Jackson2JsonRedisSerializer jackson2JsonRedisSerializer=new
Jackson2JsonRedisSerializer(Object.class);
ObjectMapper objectMapper=new ObjectMapper();
objectMapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY);
objectMapper.activateDefaultTyping(LaissezFaireSubTypeValidator.instance);
//string的序列化
StringRedisSerializer stringRedisSerializer=new StringRedisSerializer();
//key采纳string的序列化形式
template.setKeySerializer(stringRedisSerializer);
//value采纳jackson的序列化形式
template.setValueSerializer(jackson2JsonRedisSerializer);
//hashkey采纳string的序列化形式
template.setHashKeySerializer(stringRedisSerializer);
//hashvalue采纳jackson的序列化形式
template.setHashValueSerializer(jackson2JsonRedisSerializer);
template.afterPropertiesSet();
return template;
}
}
最初是controller
@RestController
@RequestMapping("/token")
public class TokenController {
@Autowired
private TokenService tokenService;
@GetMapping
public Response token(){
return tokenService.createToken();
}
@PostMapping("checktoken")
public Response checktoken(HttpServletRequest request){
return tokenService.checkToken(request);
}
}
其余代码在文末github链接上自取
(四)后果验证
首先通过token接口创立一个token进去,此时redis中也存在了改token
在jmeter中同时运行50个申请,咱们能够察看到,只有第一个申请校验胜利,后续的申请均提醒请勿反复操作。
jmeter压测文件(Token Plan.jmx)和代码自取:github自取
发表回复