基于SpringBoot的自定义注解

基于SpringBoot的自定义注解

Java注解

Java 定义的注解分三类。

(1)普通注解。
(2)元注解。
(3)自定义注解。

1、普通注解

普通注解在Java.lang 中有3个:

  • @Override:检查该方法是否是重写方法。如果发现其父类,或者是引用的接口中并没有该方法时,会报编译错误。
  • @Deprecated:标记过时方法。若某类或某方法加上该注解之后,表示此方法或类不再建议使用,调用时也会出现删除线,但并不代表不能用,只是说,不推荐使用,因为还有更好的方法可以调用。
  • @SuppressWarnings:指示编译器去忽略注解中声明的警告。

2、元注解

元注解就是用来修饰注解的注解。通俗的理解为为了开发人员方便开发自定义注解的工具型注解,简单说的就类似JDK工具包作用。

在java.lang.annotation中有4个元注解:

  • @Retention:用来说明注解的存活时间,是只在代码中,还是编入class文件中,或者是在运行时可以通过反射访问。保存方式有3种:
    • RetentionPolicy.SOURCE - 标记的注释仅保留在源级别中,并由编译器忽略
    • RetentionPolicy.CLASS - 标记的注释在编译时由编译器保留,但Java虚拟机(JVM)会忽略
    • RetentionPolicy.RUNTIME - 标记的注释由JVM保留,因此运行时环境可以使用它
  • @Documented:标记这些注解是否包含在用户文档中,注释表明,无论何时使用指定的注释,都应使用Javadoc工具记录这些元素。(默认情况下,注释不包含在Javadoc中)
  • @Target:标记这个注解应该是哪种 Java 对象范围,所修饰的对象范围有:Annotation可被用于 packages、types(类、接口、枚举、Annotation类型)、类型成员(方法、构造方法、成员变量、枚举值)、方法参数和本地变量(如循环变量、catch参数),支持的元素类型有:
    • ElementType.TYPE 可以应用于类的任何元素
    • ElementType.FIELD 可以应用于字段或属性
    • ElementType.METHOD 可以应用于方法级注释
    • ElementType.PARAMETER 可以应用于方法的参数
    • ElementType.CONSTRUCTOR 可以应用于构造函数
    • ElementType.LOCAL_VARIABLE 可以应用于局部变量
    • ElementType.ANNOTATION_TYPE 可以应用于注释类型
    • ElementType.PACKAGE 可以应用于包声明
    • ElementType.TYPE_PARAMETER 可以应用于类型参数
    • lementType.TYPE_USER 可以应用任何类型名称
  • @Inherited:此注释仅适用于类声明。 类继承关系中,子类会继承父类使用的注解中被@Inherited修饰的注解;接口继承关系中,子接口不会继承父接口中的任何注解,不管父接口中使用的注解有没有被@Inherited修饰;类实现接口关系中,类实现接口时不会继承任何接口中定义的注解。
    @Inherited
    @Target(ElementType.TYPE)
    @Retention(RetentionPolicy.RUNTIME)
    public @interface Person {
    }

    @Person
    public class Student {
    }

    public class ZhangSan extends Student{
    }

注解@Person@Inherited修饰,Student@Person修饰,ZhangSan继承Student(ZhangSan上又无其他注解),那么ZhangSan就会拥有Person这个注解。

3、新增注解

Java7之后增加的3个注解:

  • @SafeVarargs: Java 7 开始支持,忽略任何使用参数为泛型变量的方法或构造函数调用产生的警告。
  • @FunctionalInterface:Java 8 开始支持,标识一个匿名函数或函数式接口。
  • @Repeatable:Java 8 开始支持,标识某注解可以在同一个声明上使用多次。

3、自定义注解

开发者自己开发的注解。

实现自定义注解

环境准备

创建springboot工程,因为要测试Restful接口,需要引入spring-boot-starter-web

准备controller用来测试:

@RestController
@RequestMapping("/api")
public class ApiController {

    @RequestMapping("/checkTest01")
    public String checkTest01() {
        return "checkTest01";
    }

    @RequestMapping("/checkTest02")
    public String checkTest02() {
        return "checkTest02";
    }

    @RequestMapping("/checkTest03")
    public String checkTest03() {
        return "checkTest03";
    }
}

自定义注解

package com.pbad.springboot.annotation;

import java.lang.annotation.*;


@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD})
public @interface ApiChecked {

    /**
     * 用来区分api的名字
     * @author pbad
     * @date 下午11:35 2022/2/5
     * @return java.lang.String
     **/
    String name() default "default";

    /**
     * 用来表示当前api开启限制标记
     * @author pbad
     * @date 下午11:35 2022/2/5
     * @return boolean
     **/
    boolean limit() default true;

    /**
     * 用来表示当前api等待时间
     * @author pbad
     * @date 下午11:36 2022/2/5
     * @return long
     **/
    long waitingTime() default 10;
}

改造Controller接口

@RestController
@RequestMapping("/api")
public class ApiController {

    /**
     * 测试接口1,设置limit = false,其余为默认
     * @author pbad
     * @date 下午10:45 2022/2/6
     * @return java.lang.String
     **/
    @RequestMapping("/checkTest01")
    @ApiChecked(limit = false)
    public String checkTest01() {
        return "checkTest01";
    }

    /**
     * 测试接口2,设置name
     * @author pbad
     * @date 下午10:46 2022/2/6
     * @return java.lang.String
     **/
    @RequestMapping("/checkTest02")
    @ApiChecked(name = "check02",limit = true)
    public String checkTest02() {
        return "checkTest02";
    }

    /**
     * 测试接口3,设置waitingTime
     * @author pbad
     * @date 下午10:46 2022/2/6
     * @return java.lang.String
     **/
    @RequestMapping("/checkTest03")
    @ApiChecked(name = "check03",limit = true,waitingTime = 60)
    public String checkTest03() {
        return "checkTest03";
    }
}

拦截器解析注解

自定义拦截器

package com.pbad.springboot.intercepter;

import com.pbad.springboot.annotation.ApiChecked;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;
import org.springframework.web.method.HandlerMethod;
import org.springframework.web.servlet.HandlerInterceptor;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

/**
 * 拦截器解析注解
 * @author pbad
 * @date 2022年02月05日 下午11:38
 */
@Slf4j
@Component
public class ApiCheckedIntercepter implements HandlerInterceptor{

    /**
     * 前置拦截器
     * @author pbad
     * @date 下午11:43 2022/2/5
     * @param request
     * @param response
     * @param handler
     * @return boolean
     **/
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        log.info(" into intercepter");
        //注解相关数据验证
        if (!apiChecked(response,handler)){
            return false;
        }
        return true;
    }

    /**
     * 拦截器方式使用注解,解析注解参数
     * @author pbad
     * @date 下午11:47 2022/2/5 
     * @param response
     * @param handler 
     * @return boolean
     **/
    private boolean apiChecked(HttpServletResponse response, Object handler) throws Exception{
        // 反射获取方法上的LoginRequred注解
        HandlerMethod handlerMethod = (HandlerMethod)handler;
        ApiChecked interfaceChecked = handlerMethod.getMethod().getAnnotation(ApiChecked.class);
        
        // 注解为空,返回为true
        if(interfaceChecked == null){
            log.info("preHandle interfaceChecked is null ");
            return true;
        }
        
        // 注解是否开启限制(默认开启)
        if (interfaceChecked.limit()){
            //执行业务逻辑....
            //这里直接输出注解参数,不实现具体的业务逻辑
            System.out.println(interfaceChecked.name());
            System.out.println(interfaceChecked.limit());
            System.out.println(interfaceChecked.waitingTime());
        }
        return true;
    }
}

将自定义拦截器加入SpringBoot配置中去

/**
 * @author pbad
 * @date 2022年02月05日 下午11:54
 */
@Configuration
public class InterceptorTrainConfig implements WebMvcConfigurer {

    /**
     * 拦截器注册类 InterceptorRegistry 加入自己的拦截器
     * @param registry
     */
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        // 加入自己的拦截器,针对/api下的所有请求
        registry.addInterceptor(new ApiCheckedIntercepter()).addPathPatterns("/api/**");
    }
}

注解测试

启动项目,访问http://localhost:8080/api/checkTest01,测试接口1

output:

checkTest01

Console:

访问http://localhost:8080/api/checkTest02,测试接口2

output:

checkTest01

Console:
check02
true
10

访问http://localhost:8080/api/checkTest03,测试接口3

output:

Aop实现

使用Aop实现时,请先将拦截器配置移除,避免影响测试结果

引入AOP

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-aop</artifactId>
        </dependency>

AOP切面类

package com.pbad.springboot.aspect;

import com.pbad.springboot.annotation.ApiChecked;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;

import java.util.Arrays;

/**
 * 切面类
 * @author pbad
 * @date 2022年02月06日 下午10:20
 */
@Slf4j
@Aspect
@Component
public class LimitCheckedAspect {

    @Order(1) // Order 代表优先级,数字越小优先级越高
    //定义切入点
    @Pointcut("@annotation(com.pbad.springboot.annotation.ApiChecked)")
    public void checkedPoint(){};

    /**
     * 环绕通知,针对注解ApiChecked进行处理
     * @author pbad
     * @date 下午10:57 2022/2/6
     * @param joinPoint
     * @param apiChecked
     * @return java.lang.Object
     **/
    @Around(value = "checkedPoint() && @annotation(apiChecked)")
    public Object doLogs(ProceedingJoinPoint joinPoint, ApiChecked apiChecked) throws Throwable {
        // 获取方法名称
        String methodName = joinPoint.getSignature().getName();
        // 获取入参
        Object[] param = joinPoint.getArgs();
        System.out.println("方法名称:"+methodName);
        System.out.println("参数");
        for (Object o : param) {
            System.out.println(o);
        }
        log.debug("methodName {}, param {}", methodName, param);

        // 注解拦截处理
        System.out.println(apiChecked.name());
        System.out.println(apiChecked.limit());
        System.out.println(apiChecked.waitingTime());

        //执行原有接口方法
        Object result = null;
        result = joinPoint.proceed();
        return result;
    }
}

注解测试

启动项目,访问http://localhost:8080/api/checkTest01,测试接口1

output:

checkTest01

Console:
方法名称:checkTest01
参数
default
false
10

访问http://localhost:8080/api/checkTest02,测试接口2

output:

checkTest02

Console:
方法名称:checkTest02
参数
check02
true
10

访问http://localhost:8080/api/checkTest03,测试接口3

output:

checkTest02

Console控制台:
方法名称:checkTest03
参数
check03
true
60

参考链接

上一篇:springboot实战小项目-前端-文件代码该删删


下一篇:QLineEdit 编辑框左侧添加小图标