使用aop、解析器处理自定义注解

本文介绍了如何在Java中使用AOP进行前置通知处理自定义注解,主要用于限制字段长度。由于无法在参数级别通过AOP拦截,选择了使用解析器来处理加在参数和属性上的注解。对于多个参数的情况,选择了不处理,而当属性添加注解时,不限定特定场景,所有带有注解的属性都会进行长度限制。同时,配置了解析器,并利用全局异常处理进行错误反馈。欢迎大家提供更好的实现方式。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

先随便定义一个自定义注解

@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 {

	/**
     * 处理器,这里是在处理参数、属性上的LimitLength注解
     *
     * @date 2022-04-26 10:44
     * @param methodParameter methodParameter
     * @return  是否处理
     */
    @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;
    }

	/**
     * 处理器,这里是在处理参数、属性上的LimitLength注解
     *
     * @date 2022-04-26 10:44
     * @param methodParameter methodParameter
     * @param modelAndViewContainer modelAndViewContainer
     * @param nativeWebRequest nativeWebRequest
     * @param webDataBinderFactory webDataBinderFactory
     * @return  参数
     */
    @Override
    public Object resolveArgument(MethodParameter methodParameter, ModelAndViewContainer modelAndViewContainer, NativeWebRequest nativeWebRequest, WebDataBinderFactory webDataBinderFactory) throws Exception {
        HttpServletRequest request = nativeWebRequest.getNativeRequest(HttpServletRequest.class);
        // HttpServletResponse response = nativeWebRequest.getNativeResponse(HttpServletResponse.class);

        // 获取所有参数的类型,这里处理的是注解在属性上时的情况
        Class<?> parameterType = methodParameter.getParameterType();
        Field[] fields = parameterType.getDeclaredFields();
       	for (Field field : fields) {
            LimitLength annotation = field.getAnnotation(LimitLength.class);
            if (ObjectUtil.isNotEmpty(annotation)) {
                // 如果存在这个注解,则直接从入参里获取这个param
                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)) {
        	// 如果存在这个注解,则直接从入参里获取这个param
            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的,当然你用其他的帮助类或者手动判空也行。

可能有很多地方写的会比较乱,某些地方可能有更好的获取注解、属性、参数的方法,欢迎大佬指正(诶嘿)。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值