SpringBoot中的自定义注解开发与注解处理器

简介: 使用AnnotatedElement接口中的方法提取注解中的数据,像Class/Constructor/Field/Method/Package这些类都实现了AnnotatedElement接口 注:只有当定义Annotation时使用了@Retention(RetentionPolicy.RUNTIME)修饰, JVM才会在装载class文件时提取保存在class文件中的Annotation,该Annotation才会在运行时可见,这样我们才能够解析
  1. Java注解(Annotation)
    Java注解是附加在代码中的一些元信息,用于一些工具在编译、
    运行时进行解析和使用,起到说明、配置的功能。

    注解相关类都包含在java.lang.annotation包中。

  2. Java注解分类
    2.1 JDK基本注解
    2.2 JDK元注解
    2.3 自定义注解
  3. JDK基本注解
    3.1 @Override

    重写

    3.2 @Deprecated

    已过时 

    3.3 @SuppressWarnings(value = "unchecked")

    压制编辑器警告
    
    
  4. JDK元注解
    元注解用于修饰其他的注解(纪委:管干部的干部)
    4.1 @Retention:定义注解的保留策略

    @Retention(RetentionPolicy.SOURCE)             //注解仅存在于源码中,在class字节码文件中不包含
    @Retention(RetentionPolicy.CLASS)              //默认的保留策略,注解会在class字节码文件中存在,但运行时无法获得,
    @Retention(RetentionPolicy.RUNTIME)            //注解会在class字节码文件中存在,在运行时可以通过反射获取到
    

    4.2 @Target:指定被修饰的Annotation可以放置的位置(被修饰的目标)

    @Target(ElementType.TYPE)                      //接口、类
    @Target(ElementType.FIELD)                     //属性
    @Target(ElementType.METHOD)                    //方法
    @Target(ElementType.PARAMETER)                 //方法参数
    @Target(ElementType.CONSTRUCTOR)               //构造函数
    @Target(ElementType.LOCAL_VARIABLE)            //局部变量
    @Target(ElementType.ANNOTATION_TYPE)           //注解
    @Target(ElementType.PACKAGE)                   //包 
    
    注:可以指定多个位置,例如:

@Target({ElementType.METHOD, ElementType.TYPE}),也就是此注解可以在方法和类上面使用

4.3 @Inherited:指定被修饰的Annotation将具有继承性

4.4 @Documented:指定被修饰的该Annotation可以被javadoc工具提取成文档.

  1. 注解分类
    根据Annotation是否包含成员变量,可以把Annotation分为两类:
    5.1 标记Annotation:

没有成员变量的Annotation; 这种Annotation仅利用自身的存在与否来提供信息
5.2 元数据Annotation:

  包含成员变量的Annotation; 它们可以接受(和提供)更多的元数据;

自定义注解开发
使用@interface关键字, 其定义过程与定义接口非常类似, 需要注意的是:
Annotation的成员变量在Annotation定义中是以无参的方法形式来声明的, 其方法名和返回值类型定义了该成员变量的名字和类型,
而且我们还可以使用default关键字为这个成员变量设定默认值,例如:

@Inherited

@Retention(RetentionPolicy.RUNTIME)

@Target({ElementType.METHOD, ElementType.TYPE})

public @interface Tag {

 String name() default "该叫啥才好呢?";


 String description() default "这家伙很懒, 啥也没留下...";

}

注1:只有名字为“value”属性,赋值时可以省略属性名

  1. 提取Annotation信息
    使用AnnotatedElement接口中的方法提取注解中的数据,像Class/Constructor/Field/Method/Package这些类都实现了AnnotatedElement接口
    注:只有当定义Annotation时使用了@Retention(RetentionPolicy.RUNTIME)修饰,

     JVM才会在装载class文件时提取保存在class文件中的Annotation,该Annotation才会在运行时可见,这样我们才能够解析
    

    案例一:提供类的注解
    案例二:属性上的注解
    案例三:方法上的注解

  2. 注解处理器
    8.1 使用Annotation修饰了类/方法/成员变量等之后,这些Annotation不会自己生效,为了让程序中的这些注解起作用,

    必须由这些注解的开发者为这些注解提供一个注解处理器(Annotation Processor)。另外,在编译期间,
    JVM会自动运行注解处理器(当然,我们需要将其注册)
    

    8.2 创建注解处理器

    自定义处理器都需要继承于AbstractProcessor
    init(ProcessingEnvironment processingEnv):每一个注解处理器类都必须有一个空的构造函数。init()方法会被注解处理工具调用,并输入ProcessingEnviroment参数。ProcessingEnviroment提供很多有用的工具类Elements, Types和Filer。
    process(Set<? extends TypeElement> annotations, RoundEnvironment 

env):这是最重要的一个方法,你在这里写你的扫描、评估和处理注解的代码,以及生成Java文件。输入参数RoundEnviroment,可以让你查询出包含特定注解的被注解元素

  
  getSupportedAnnotationTypes():该注解处理器是注册到哪些注解上。它的返回值是一个字符串的集合,包含本处理器想要处理的注解类型的合法全称。    
  getSupportedSourceVersion():用来指定你使用的Java版本。通常这里返回SourceVersion.latestSupported()。然而,如果你有足够的理由只支持Java 7的话,你也可以返回SourceVersion.RELEASE_7。

  注1:在Java7还可以使用如下两个注解,而不用重写getSupportedAnnotationTypes()和getSupportedSourceVersion()
  @SupportedSourceVersion(SourceVersion.latestSupported())
  @SupportedAnnotationTypes({// 合法注解全名的集合})

  但要考虑代码兼容性的原因,特别是针对Android平台,我建议使用重载getSupportedAnnotationTypes()
  和getSupportedSourceVersion()方法代替@SupportedAnnotationTypes和@SuppozrtedSourceVersion

  注2:init方法参数详解:ProcessingEnvironment
       Elements elementUtils = processingEnv.getElementUtils();//一个用来处理Element的工具类;
       Types typeUtils = processingEnv.getTypeUtils();         //一个用来处理TypeMirror的工具类;
       Filer filer = processingEnv.getFiler();                 //正如这个名字所示,使用Filer你可以创建文件;       
       Messager messager = processingEnv.getMessager();        //一个用来处理消息的工具类,为了输出错误信息。
                                                               //因为在注解处理器里面不可以抛出Exception!
                                                               //为什么了?因为在编译阶段抛出错误后,注解处理器就不会运行完,
                                                               //也就没什么用了。 所以Message就是为了输出错误信息 

       private void error(Element e, String msg, Object... args) {
         messager.printMessage(
            Diagnostic.Kind.ERROR,
            String.format(msg, args),
            e);
       }

       
  注3:Element代表的是源代码,它的子类有这些:
       PackageElement:包名
       TypeElement:类
       VariableElement:变量
       ExecutableElement:方法
       详细见资料“Car.java”
      
  注4:process方法返回true即是退出处理
       return true; // 退出处理
       
       

8.3 注册注解处理器(静态处理器,非AOP处理)

8.3.1 手动
      在当前项目中的resources/META-INF/services目录需要新建一个特殊的文件javax.annotation.processing.Processor
      javax.annotation.processing.Processor文件的内容是注解处理器的合法的全名列表,每一个元素换行分割
      - META-INF 
      - - services 
      - - - javax.annotation.processing.Processor           

8.3.2 使用google的auto-service项目
      1. 导入依赖
         <dependency>
           <groupId>com.google.auto.service</groupId>
           <artifactId>auto-service</artifactId>
           <version>1.0-rc4</version>
         </dependency>
      2. 在注解处理器上添加@AutoService(Processor.class)即可

8.4 Maven打包(打包涵依赖jar包 )

   MAVEN打AR包,注意还要将依赖打到Jar包中。详情见资料“test1-pom.xml”


  1. 动态注解处理器(spring aop方式)

附录一:本章英语单词
Retention:保留
Policy:策略
RUNTIME:运行时间
Annotation:注释

@Service用于标注业务层组件

@Controller用于标注控制层组件(如struts中的action)

@Repository用于标注数据访问组件,即DAO组件

@Component泛指组件,当组件不好归类的时候,我们可以使用这个注解进行标注。

@Aspect//切面

@Pointcut//定义需要切面的地方,表达式参数(https://blog.csdn.net/elim168/article/details/78150438

@annotation//当执行的方法上拥有指定的注解时生效。

@After

@Before

@Around

相关文章
|
2月前
|
缓存 监控 Java
SpringBoot @Scheduled 注解详解
使用`@Scheduled`注解实现方法周期性执行,支持固定间隔、延迟或Cron表达式触发,基于Spring Task,适用于日志清理、数据同步等定时任务场景。需启用`@EnableScheduling`,注意线程阻塞与分布式重复问题,推荐结合`@Async`异步处理,提升任务调度效率。
494 128
|
2月前
|
XML 安全 Java
使用 Spring 的 @Aspect 和 @Pointcut 注解简化面向方面的编程 (AOP)
面向方面编程(AOP)通过分离横切关注点,如日志、安全和事务,提升代码模块化与可维护性。Spring 提供了对 AOP 的强大支持,核心注解 `@Aspect` 和 `@Pointcut` 使得定义切面与切入点变得简洁直观。`@Aspect` 标记切面类,集中处理通用逻辑;`@Pointcut` 则通过表达式定义通知的应用位置,提高代码可读性与复用性。二者结合,使开发者能清晰划分业务逻辑与辅助功能,简化维护并提升系统灵活性。Spring AOP 借助代理机制实现运行时织入,与 Spring 容器无缝集成,支持依赖注入与声明式配置,是构建清晰、高内聚应用的理想选择。
402 0
|
2月前
|
Java 测试技术 API
将 Spring 的 @Embedded 和 @Embeddable 注解与 JPA 结合使用的指南
Spring的@Embedded和@Embeddable注解简化了JPA中复杂对象的管理,允许将对象直接嵌入实体,减少冗余表与连接操作,提升数据库设计效率。本文详解其用法、优势及适用场景。
285 126
|
3月前
|
XML JSON Java
Spring框架中常见注解的使用规则与最佳实践
本文介绍了Spring框架中常见注解的使用规则与最佳实践,重点对比了URL参数与表单参数的区别,并详细说明了@RequestParam、@PathVariable、@RequestBody等注解的应用场景。同时通过表格和案例分析,帮助开发者正确选择参数绑定方式,避免常见误区,提升代码的可读性与安全性。
|
1月前
|
XML Java 应用服务中间件
【SpringBoot(一)】Spring的认知、容器功能讲解与自动装配原理的入门,带你熟悉Springboot中基本的注解使用
SpringBoot专栏开篇第一章,讲述认识SpringBoot、Bean容器功能的讲解、自动装配原理的入门,还有其他常用的Springboot注解!如果想要了解SpringBoot,那么就进来看看吧!
336 2
|
2月前
|
XML Java 数据格式
常用SpringBoot注解汇总与用法说明
这些注解的使用和组合是Spring Boot快速开发和微服务实现的基础,通过它们,可以有效地指导Spring容器进行类发现、自动装配、配置、代理和管理等核心功能。开发者应当根据项目实际需求,运用这些注解来优化代码结构和服务逻辑。
285 12
|
2月前
|
Java 测试技术 数据库
使用Spring的@Retryable注解进行自动重试
在现代软件开发中,容错性和弹性至关重要。Spring框架提供的`@Retryable`注解为处理瞬时故障提供了一种声明式、可配置的重试机制,使开发者能够以简洁的方式增强应用的自我恢复能力。本文深入解析了`@Retryable`的使用方法及其参数配置,并结合`@Recover`实现失败回退策略,帮助构建更健壮、可靠的应用程序。
308 1
使用Spring的@Retryable注解进行自动重试
|
7月前
|
前端开发 Java 关系型数据库
基于Java+Springboot+Vue开发的鲜花商城管理系统源码+运行
基于Java+Springboot+Vue开发的鲜花商城管理系统(前后端分离),这是一项为大学生课程设计作业而开发的项目。该系统旨在帮助大学生学习并掌握Java编程技能,同时锻炼他们的项目设计与开发能力。通过学习基于Java的鲜花商城管理系统项目,大学生可以在实践中学习和提升自己的能力,为以后的职业发展打下坚实基础。技术学习共同进步
503 7
|
2月前
|
传感器 Java 数据库
探索Spring Boot的@Conditional注解的上下文配置
Spring Boot 的 `@Conditional` 注解可根据不同条件动态控制 Bean 的加载,提升应用的灵活性与可配置性。本文深入解析其用法与优势,并结合实例展示如何通过自定义条件类实现环境适配的智能配置。
170 0
探索Spring Boot的@Conditional注解的上下文配置

热门文章

最新文章