• 自定义注解使用


    注解介绍

    它提供了一种安全的类似注释的机制,用来将任何的信息或元数据(metadata)与程序元素(类、方法、成员变量等)进行关联。为程序的元素(类、方法、成员变量)加上更直观、更明了的说明,这些说明信息是与程序的业务逻辑无关,并且供指定的工具或框架使用。Annontation像一种修饰符一样,应用于包、类型、构造方法、方法、成员变量、参数及本地变量的声明语句中

    注解本质是一个继承了Annotation 的特殊接口,其具体实现类是Java 运行时生成的动态代理类。而我们通过反射获取注解时,返回的是Java 运行时生成的动态代理对象$Proxy1。通过代理对象调用自定义注解(接口)的方法,会最终调用AnnotationInvocationHandler 的invoke 方法 

    元注解

    所有元注解定义在java.lang.annotation包下面,其中Annotation是注解的基本接口,所有的注解都继承这个接口

     java.lang.annotation 提供了四种元注解,专门注解其他的注解(在自定义注解的时候,需要使用到元注解)

    1、@Documented:指定被标注的注解会包含在javadoc中

    2、@Retention: 指定注解的生命周期(源码、class文件、运行时),其参考值见类的定义:java.lang.annotation.RetentionPolicy

    RetentionPolicy.SOURCE : 在编译阶段丢弃。这些注解在编译结束之后就不再有任何意义,所以它们不会写入字节码。@Override, @SuppressWarnings都属于这类注解。
    RetentionPolicy.CLASS : 在类加载的时候丢弃。在字节码文件的处理中有用。注解默认使用这种方式。
    RetentionPolicy.RUNTIME : 始终不会丢弃,运行期也保留该注解,因此可以使用反射机制读取该注解的信息。我们自定义的注解通常使用这种方式。 

    3、@Target:指定注解使用的目标范围(类、方法、字段等),其参考值见类的定义:java.lang.annotation.ElementTyp

    ElementType.CONSTRUCTOR :用于描述构造器。

    ElementType.FIELD :成员变量、对象、属性(包括enum实例)。
    ElementType.LOCAL_VARIABLE: 用于描述局部变量。
    ElementType.METHOD : 用于描述方法。
    ElementType.PACKAGE :用于描述包。
    ElementType.PARAMETER :用于描述参数。
    ElementType.ANNOTATION_TYPE:用于描述参数
    ElementType.TYPE :用于描述类、接口(包括注解类型) 或enum声明。

     4、@Inherited:指定子类可以继承父类的注解,只能是类上的注解,方法和字段的注解不能继承。即如果父类上的注解是@Inherited修饰的就能被子类继承

    注解处理器类库 

    AnnotatedElement 接口是所有程序元素(Class、Method和Constructor)的父接口,所以程序通过反射获取了某个类的AnnotatedElement对象之后,程序就可以调用该对象的如下方法来访问Annotation信息:

      方法1:  T  getAnnotation(Class annotationClass): 返回改程序元素上存在的、指定类型的注解,如果该类型注解不存在,则返回null。
      方法2:Annotation[]  getAnnotations():返回该程序元素上存在的所有注解。
      方法3:boolean  isAnnotationPresent(Class annotationClass):判断该程序元素上是否包含指定类型的注解,存在则返回true,否则返回false.
      方法4:Annotation[]  getDeclaredAnnotations(Class annotationClass):返回直接存在于此元素上的所有注释。与此接口中的其他方法不同,该方法将忽略继承的注释。(如果没有注释直接存在于此元素上,则返回长度为零的一个数组。)该方法的调用者可以随意修改返回的数组;这不会对其他调用者返回的数组产生任何影响

    代码案例:

    1. public extends Serializable> List generateLog(T target) {
    2. Classextends Serializable> clz = target.getClass();
    3. Field[] fields = clz.getDeclaredFields();
    4. Method[] methods = clz.getMethods();
    5. SystemLogging clzAnnotation = null;
    6. // 类上是否有标注了@systemlogging注解
    7. if(clz.isAnnotationPresent(SystemLogging.class)){
    8. }
    9. for (Method method : methods) {
    10. // 方法上是否标注了@systemlogging注解
    11. if(method.isAnnotationPresent(SystemLogging.class)){
    12. }
    13. }
    14. for(Field field : fields){
    15. // 属性上是否有标注了@systemlogging注解
    16. if(field.isAnnotationPresent(SystemLogging.class)){
    17. }
    18. // 获取属性上指定注解的内容,不存在返回null
    19. SystemLogging systemLogging = field.getAnnotation(SystemLogging.class);
    20. }
    21. }

    自定义注解

    • Annotation 类型定义为@interface, 所有的Annotation 会自动继承java.lang.Annotation这一接口,并且不能再去继承别的类或是接口。
    •   参数成员只能用public 或默认(default) 这两个访问权修饰。语法:类型  属性名()  [default 默认值];      default表示默认值 ,也可以不编写默认值的.
    •   参数成员只能用基本类型byte、short、char、int、long、float、double、boolean八种基本数据类型和String、Enum、Class、annotations等数据类型,以及这一些类型的数组.
    •   要获取类方法和字段的注解信息,必须通过Java的反射技术来获取 Annotation 对象,因为你除此之外没有别的获取注解对象的方法。
    •   注解也可以没有定义成员,,不过这样注解就没啥用了

    代码案例

    1. @Target({ElementType.FIELD, ElementType.TYPE})
    2. @Retention(RetentionPolicy.RUNTIME)
    3. @Documented
    4. public @interface SystemLogging {
    5. String[] keys() default {};
    6. String value() default "";
    7. String secondValue() default "";
    8. String keyName() default "字段";
    9. Classextends KeyEnums>[] keyEnum() default {};
    10. boolean deleted() default false;
    11. }

    springboot中的自定义注解类

    springboot扩展注解:SpringBoot重点详解--@Conditional注解_pengjunlee的博客-CSDN博客

  • 相关阅读:
    无胁科技-TVD每日漏洞情报-2022-11-21
    2023年高教社杯数学建模国赛 赛题浅析
    leetcode:415. 字符串相加(模拟竖式计算)
    怎么保护公司文件安全
    【BOOST C++ 5 】通信(03 网络编程 )
    kafka的auto.offset.reset详解与测试
    交联聚苯乙烯微球固载9氮杂二环[3,3,1]壬9氧自由基/聚苯乙烯-二乙烯基苯微球性能解析
    [Spring Cloud] nacos作为服务中心
    马士兵-郑金维—并发编程—4.阻塞队列
    在WSL2中安装多个Ubuntu实例
  • 原文地址:https://blog.csdn.net/qq_36042938/article/details/126713109