首页 > 编程语言 >Java自定义Annotation注解开发详解

Java自定义Annotation注解开发详解

时间:2022-09-03 00:44:21浏览次数:87  
标签:Processor Java 自定义 class Immutable Annotation 注解 public

Java自定义Annotation注解开发详解

目录

介绍

一、运行期的自定义注解

1. Class Level Annotation

2. Method Level Annotation

3. Field Level Annotation

4. 使用自定义注解

5. 处理自定义注解的逻辑

二、编译期的自定义注解

1. 创建自定义注解

2. 实现一个Processor

3. 注册你的Processor

4. 测试你的自定义注解


介绍

Java中的注解是每个开发都会遇到的,但是如果要自定义自己的注解,则需要遵循一些基本的步骤,一般注解的开发有2个基本方法:

  1. 在运行期,通过反射获得当前类,方法,变量上的注解信息来实现自定义注解的功能
  2. 在编译期,通过Annotation Processer预编译生成想要的任何内容或者逻辑

下面将通过2个例子来说明开发一个自定义注解需要哪些步骤。首先我们将看到一个非常简单的例子,我们用这个例子来说明开发自定义annotation的一些基本步骤,我们的第二个例子将介绍自定义注解以及Annotation Processor的一些用法

一、运行期的自定义注解

在下面的例子中,我们将创建3种不同类型自定义注解,以收集所有有自定义注解的类和方法

1. Class Level Annotation

  1. @Retention(RetentionPolicy.RUNTIME)
  2. @Target(ElementType.TYPE)
  3. public @interface ClassAnnotation {
  4. public String alias() default "";
  5. }

2. Method Level Annotation

  1. @Retention(RetentionPolicy.RUNTIME)
  2. @Target(ElementType.METHOD)
  3. public @interface MethodAnnotation {
  4. public String alias() default "";
  5. }

3. Field Level Annotation

  1. @Retention(RetentionPolicy.RUNTIME)
  2. @Target(ElementType.FIELD)
  3. public @interface FieldAnnotation {
  4. public String alias() default "";
  5. }

@Retention注解解释:

  • @Retention(RetentionPolicy.SOURCE): 该注解只在编译期生效,生成的class文件并不包含该注解
  • @Retention(RetentionPolicy.CLASS): 该注解会被保留在class文件中,但是运行期不会生效
  • @Retention(RetentionPolicy.RUNTIME): 该注解会被保留在class文件中,并且会在运行期生效

@Target注解解释如下,这里只列举了部分@Target 类型,更多的类型请参看JavaDoc。

  • @Target(ElementType.TYPE): 该注解只能运用到Class, Interface, enum上
  • @Target(ElementType.FIELD): 该注解只能运用到Field上
  • @Target(ElementType.METHOD): 该注解只能运用到方法上

注解中还有一个alias的string类型参数,缺省值是空字符串,在下一节我们将看到如何使用这个string类型的参数

4. 使用自定义注解

  1. @ClassAnnotation(alias = "test")
  2. public class Test {
  3. @FieldAnnotation
  4. private String name;
  5. @MethodAnnotation(alias="debug")
  6. public String getName() {
  7. return name;
  8. }
  9. }

我们在class, field, method上分别运用我们的自定义注解,并且在method上开启debug日志

5. 处理自定义注解的逻辑

我们已经介绍了如何定义自己的注解,以及如何使用我们的注解,接下来我们将用Java的Reflection API来实现我们自定义注解的逻辑

  1. public void gatherAnnotations {
  2. Map<String, Class> classMap = new HashMap<>();
  3. Class<Test> obj = Test.class;
  4. if(obj.isAnnotationPresent(ClassAnnotation.class)) {
  5. ClassAnnotation classAnnotation = obj.getAnnotation(ClassAnnotation.class);
  6. if ("".equals(classAnnotation.alias())) {
  7. classMap.put(obj.getName(), obj);
  8. }else{
  9. classMap.put(classAnnotation.alias(), obj);
  10. }
  11. }
  12. Map<String, Method> methodMap = new HashMap<>();
  13. for (Method method : obj.getDeclaredMethods()) {
  14. if (method.isAnnotationPresent(MethodAnnotation.class)) {
  15. MethodAnnotation methodAnnotation = method.getAnnotation(MethodAnnotation.class);
  16. if ("".equals(methodAnnotation.alias())) {
  17. methodMap.put(method.getName(), method);
  18. }else{
  19. methodMap.put(methodAnnotation.alias(), method);
  20. }
  21. }
  22. }
  23. }

这里我们运用Java反射收集了所有有我们自定义注解的类和方法,并放到相应的Map中。

至此,我们完成了一个简单的运行期自定义注解的例子,这个例子看上去没有实际的用处,但是在真正的业务场景中,有很多应用都是基于此类逻辑,例如Spring中的@Service和@Autowired注解大都基于这样的逻辑,来进行后续的初始化和注入。

二、编译期的自定义注解

Annotation Processor是代码级别的注解处理器,所以它一般在编译期帮助我们生成我们想要的动态代码,配置文件,文档等,它的使用场景相当广泛,一般包含以下几种,

  • 生成代码或者properties文件
  • 修改源文件,例如为Pojo生成getter和setter方法
  • 一些源文件分析诊断的案例,本文就属于这一类

在这个例子中,我们将创建一个Immutable的类级别的注解,该注解将在编译期检查class中所有的field是否有final关键字修饰

1. 创建自定义注解

  1. @Target(ElementType.TYPE)
  2. @Retention(RetentionPolicy.SOURCE)
  3. public @interface Immutable {
  4. }
  1. 定义了一个Immutable注解
  2. @Target(ElementType.TYPE)表示Immutable注解只能放在类上
  3. @Retention(RetentionPolicy.SOURCE)表示Immutable注解只在编译期生效

2. 实现一个Processor

JDK中已经为我们实现了一个AbstractProcessor, 所以我们要做的是扩展这个Abstract类,并且实现里面的process方法

  1. @SupportedAnnotationTypes("annotation.Immutable")
  2. @SupportedSourceVersion(SourceVersion.RELEASE_11)
  3. @AutoService(Processor.class)
  4. public class ImmutableProcessor extends AbstractProcessor {
  5. @Override
  6. public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
  7. List<String> nonPublicFields = new LinkedList<>();
  8. for ( Element element : roundEnv.getElementsAnnotatedWith(Immutable.class)) {
  9. if( element instanceof TypeElement ) {
  10. TypeElement typeElement = (TypeElement) element;
  11. for( final Element enclosedElement: typeElement.getEnclosedElements() ) {
  12. if( enclosedElement instanceof VariableElement) {
  13. VariableElement variableElement = ( VariableElement )enclosedElement;
  14. if( !variableElement.getModifiers().contains( Modifier.FINAL ) ) {
  15. nonPublicFields.add(variableElement.getSimpleName().toString());
  16. }
  17. }
  18. }
  19. if (nonPublicFields.size() > 0) {
  20. processingEnv.getMessager().printMessage( Diagnostic.Kind.ERROR,
  21. String.format( "Class %s is not @Immutable, fields %s are not declared as final",
  22. typeElement.getSimpleName(), String.join(",", nonPublicFields)
  23. )
  24. );
  25. }
  26. }
  27. }
  28. return true;
  29. }
  30. }
  1. @SupportedAnnotationTypes("annotation.Immutable")表示ImmutableProcessor 只用于annotation.Immutable注解
  2. @SupportedSourceVersion(SourceVersion.RELEASE_11)表示这个processor支持的JDK最低版本是11
  3. @AutoService(Processor.class)表示使用Google auto-service library注册这个processor,下一节将讨论如何注册你的processor
  4. process方法轮训找到所有有Immutable注解的类,然后遍历所有的方法,查找是否有final关键字,如果没有记录该方法,最后抛出异常

3. 注册你的Processor

Java实际上提供了好几种选择来帮助我们注册自己的Processor使我们的自定义注解生效,这里我们只介绍最常用的方法来注册Processor

  1. 通过Google auto-service library来注册你的processor
  1. @AutoService(Processor.class)
  2. public class ImmutableProcessor extends AbstractProcessor {
  3. // ...
  4. }
  • 首先你需要引入auto-service library
  • 在你的processor上加上@AutoService(Processor.class)注解
  • 编译后,你将在Jar包的META-INF/services下看到javax.annotation.processing.Processor文件,文件里包含了你的processor

      2. 通过Maven plugin来注册你的processor

       使用maven前你的processor必须已经编译,通过其他jar文件的形式加到了dependencies里

  1. <plugin>
  2. <groupId>org.apache.maven.plugins</groupId>
  3. <artifactId>maven-compiler-plugin</artifactId>
  4. <configuration>
  5. <annotationProcessors>
  6. <annotationProcessor>
  7. annotation.ImmutableProcessor
  8. </annotationProcessor>
  9. </annotationProcessors>
  10. </configuration>
  11. </plugin>

4. 测试你的自定义注解

  1. @Immutable
  2. public class Test {
  3. public String name;
  4. }

在编译期,会收到 Class Test is not @Immutable, fields name are not declared as final 的报错。至此一个简单的使用annotation processor的例子已经完成。

后续文章会继续深入分析介绍Java自定义注解在各个framework中的使用。

https://blog.csdn.net/xchann/article/details/126374273

标签:Processor,Java,自定义,class,Immutable,Annotation,注解,public
From: https://www.cnblogs.com/sunny3158/p/16651781.html

相关文章

  • [Java]《On Java》阅读记录之 -- 可变参数重载问题
    《OnJava》阅读记录之--可变参数重载问题有下面一段代码:publicclassOverloadingVarargs2{staticvoidf(floati,Character...args){System.out......
  • yml中driver-class-name: com.mysql.jdbc.Driver 解析不到的问题 java 连接数据库
    yml中driver-class-name:com.mysql.jdbc.Driver解析不到的问题java连接数据库当在idea中使用springboot的快捷创建方式时,选中了mysql和jdbc那么pom文件中会直接有......
  • JavaScript let
    JavaScriptlet1.全局作用域全局(在函数之外)声明的变量拥有全局作用域。varname1="yao";//全局作用域//此处可以使用name1functionmyFunction(){//此处也可......
  • Day01-JavaScript
    0825:Day01JS编写位置1.外链式 外部文件夹JS文件中,然后通过script标签引入 <scriptsrc="js/script.js"></script>2.嵌入式 内部的script的标签中 <script>alert("......
  • JavaScript学习
    一、什么是JavaScriptJavaScript世界上最流行的脚本语言一个合格的后端人员,必须精通JavaScript二、快速入门2.1、引入JavaScript内部引入<!DOCTYPEhtml><htmlla......
  • 网易校招-2020-正式批-Java
    编程题完美的序列我不知道为什么这样是正确的intprefectSequence(vector<int>&input){ intleft=0,right=1; longsum=input[0]; intmaxLen=0; ......
  • java复习随笔(一)
    java程序开发环境java应用程序开发离不开JDK和JRE。JDK(javadevelopmentkit),即java开发工具包,是java语言的编译环境。JDK中包含JRE。JRE(javaruntimeenvironment),即java......
  • java 并发(二)
    并发问题数据竞争死锁活锁资源不足(饿死)优先权反转数据竞争死锁四个条件资源只能互斥使用(一个资源每次只能被一个进程使用)请求者不剥夺条件(进程已获得......
  • JavaScript 一些实用辅助类库
    "usestrict";var__emptyPoint=null,__emptyContext=null;constColorRefTable=[['aliceblue','#f0f8ff'],['antiquewhite','#faebd7'],['aqua','#00......
  • Java集合---ArrayList
    集合和数组的区别共同点:都是存储数据的容器 不同点:数组的容量是固定的,集合的容量是可变的ArrayList的构造方法和添加方法publicArrayList()创建一个空......