先随便定义一个自定义注解
@Target({ElementType.METHOD, ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface LimitLength {
int minLength() default 1;
int maxLength() default 1;
}
这里预计用来限制字段的长度,本来最开始的设想是限制密码长度的(只是随便实现个功能,不要纠结加密、DTO的事),所以在加到方法上时(即下面aop处理时)仅仅处理了密码的长度。
@Aspect
@Component
public class LimitLengthAspect {
@Before("@annotation(com.islands.common.aspect.annotation.LimitLength)")
public void checkBefore(JoinPoint point) {
MethodSignature signature = (MethodSignature) point.getSignature();
Method method = signature.getMethod();
LimitLength annotation = method.getDeclaredAnnotation(LimitLength.class);
Object[] args = point.getArgs();
for (Object arg : args) {
if (arg instanceof User) {
String password = ((User) arg).getPassword();
Assert.isTrue(password.length() >= annotation.minLength() && password.length() <= annotation.maxLength(), "长度不符合标准!");
}
}
}
}
这里做了一个前置通知,因为只有加了注解的情况下会被aop拦下所以没有对注解是否存在判空。
顺带一提,这里本来的设想是把注解加到参数上,但是不知道为啥aop拦不下,所以参数是用的是解析器处理的,本来是用的拦截器,但是拦截器里的入参拿到的参数名是arg0、arg1这种的,就放弃了。
下面的是用解析器形式处理的,处理把注解加到参数、属性上的情况。多个参数用DTO做入参的情况就不处理了。加在属性上时就没有限定一定是处理密码了,只要是加了注解的属性就限制长度。因为做了全局异常捕获,所以长度不对就直接用断言抛了。
@Component
public class LimitLengthResolver implements HandlerMethodArgumentResolver {
@Override
public boolean supportsParameter(MethodParameter methodParameter) {
Class<?> parameterType = methodParameter.getParameterType();
Field[] fields = parameterType.getDeclaredFields();
for (Field field : fields) {
LimitLength annotation = field.getDeclaredAnnotation(LimitLength.class);
if (ObjectUtil.isNotEmpty(annotation)) {
return true;
}
}
Parameter methodParam = methodParameter.getParameter();
LimitLength annotation = methodParam.getDeclaredAnnotation(LimitLength.class);
ResponseBody responseBody = methodParam.getDeclaredAnnotation(ResponseBody.class);
if (ObjectUtil.isNotEmpty(annotation) && ObjectUtil.isNull(responseBody)) {
return true;
}
return false;
}
@Override
public Object resolveArgument(MethodParameter methodParameter, ModelAndViewContainer modelAndViewContainer, NativeWebRequest nativeWebRequest, WebDataBinderFactory webDataBinderFactory) throws Exception {
HttpServletRequest request = nativeWebRequest.getNativeRequest(HttpServletRequest.class);
Class<?> parameterType = methodParameter.getParameterType();
Field[] fields = parameterType.getDeclaredFields();
for (Field field : fields) {
LimitLength annotation = field.getAnnotation(LimitLength.class);
if (ObjectUtil.isNotEmpty(annotation)) {
String param = request.getParameter(field.getName());
if (ObjectUtil.isEmpty(param)) {
param = "";
}
Assert.isTrue(param.length() >= annotation.minLength() && param.length() <= annotation.maxLength(), "长度有误!");
return param;
}
}
Parameter methodParam = methodParameter.getParameter();
LimitLength annotation = methodParam.getDeclaredAnnotation(LimitLength.class);
if (ObjectUtil.isNotEmpty(annotation)) {
String param = request.getParameter(methodParam.getName());
if (ObjectUtil.isEmpty(param)) {
ResponseBody methodAnnotation = methodParameter.getMethodAnnotation(ResponseBody.class);
Assert.isNull(methodAnnotation, "LimitLength注解仅能限制单个的参数长度,请将注解加在属性上!");
param = "";
}
Assert.isTrue(param.length() >= annotation.minLength() && param.length() <= annotation.maxLength(), "长度有误!");
return param;
}
return null;
}
写一个config把解析器加上。
@Configuration
public class SpringConfig implements WebMvcConfigurer {
@Autowired
private LimitLengthResolver limitLengthResolver;
@Override
public void addArgumentResolvers(List<HandlerMethodArgumentResolver> argumentResolvers) {
argumentResolvers.add(limitLengthResolver);
}
}
悄咪咪的说一下ObjectUtil是用的hutool的,当然你用其他的帮助类或者手动判空也行。
可能有很多地方写的会比较乱,某些地方可能有更好的获取注解、属性、参数的方法,欢迎大佬指正(诶嘿)。
|