自定义注解之编译时注解(RetentionPolicy.CLASS)(二)——JavaPoet

来源:互联网 发布:网络安全法 ppt 编辑:程序博客网 时间:2024/05/15 01:09

在使用编译时注解时,需要在编译期间对注解进行处理,在这里我们没办法影响程序的运行逻辑,但我们可以进行一些需处理,比如生成一些功能性代码来辅助程序的开发,最常见的是生成.java 源文件,并在程序中可以调用到生成的文件。这样我们就可以用注解来帮助我们处理一些固定逻辑的重复性代码(如butterknife),提高开发的效率。

通过注解处理器来生成 .java 源文件基本上都会使用javapoet 这个库,JavaPoet一个是用于产生 .java 源文件的辅助库,它可以很方便地帮助我们生成需要的.java 源文件,下面来看下具体使用方法。

JavaPoet

在使用前需要先引入这个库,和 AutoService一样可以通过AndroidStudio 直接添加,如下:


下面以最简单的 HelloWorld 例子来看下怎么使用JavaPoet

先定义一个注解:

[java] view plain copy
 在CODE上查看代码片派生到我的代码片
  1. /** 
  2.  * JavaPoet HelloWorld 例子 
  3.  */  
  4. @Retention(RetentionPolicy.CLASS)  
  5. @Target(ElementType.TYPE)  
  6. public @interface JPHelloWorld {  
  7. }  

在定义个注解处理器来处理这个注解:

[java] view plain copy
 在CODE上查看代码片派生到我的代码片
  1. /** 
  2.  * 处理HelloWorld注解. 
  3.  */  
  4. @AutoService(Processor.class)  
  5. public class HelloWorldProcess extends AbstractProcessor {  
  6.   
  7.     private Filer filer;  
  8.   
  9.     @Override  
  10.     public synchronized void init(ProcessingEnvironment processingEnv) {  
  11.         super.init(processingEnv);  
  12.         // Filer是个接口,支持通过注解处理器创建新文件  
  13.         filer = processingEnv.getFiler();  
  14.     }  
  15.   
  16.     @Override  
  17.     public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {  
  18.         for (TypeElement element : annotations) {  
  19.             if (element.getQualifiedName().toString().equals(JPHelloWorld.class.getCanonicalName())) {  
  20.                 // 创建main方法  
  21.                 MethodSpec main = MethodSpec.methodBuilder("main")  
  22.                         .addModifiers(Modifier.PUBLIC, Modifier.STATIC)  
  23.                         .returns(void.class)  
  24.                         .addParameter(String[].class"args")  
  25.                         .addStatement("$T.out.println($S)", System.class"Hello, JavaPoet!")  
  26.                         .build();  
  27.                 // 创建HelloWorld类  
  28.                 TypeSpec helloWorld = TypeSpec.classBuilder("HelloWorld")  
  29.                         .addModifiers(Modifier.PUBLIC, Modifier.FINAL)  
  30.                         .addMethod(main)  
  31.                         .build();  
  32.   
  33.                 try {  
  34.                     // 生成 com.example.HelloWorld.java  
  35.                     JavaFile javaFile = JavaFile.builder("com.example", helloWorld)  
  36.                             .addFileComment(" This codes are generated automatically. Do not modify!")  
  37.                             .build();  
  38.                     // 生成文件  
  39.                     javaFile.writeTo(filer);  
  40.                 } catch (IOException e) {  
  41.                     e.printStackTrace();  
  42.                 }  
  43.             }  
  44.         }  
  45.         return true;  
  46.     }  
  47.   
  48.     @Override  
  49.     public Set<String> getSupportedAnnotationTypes() {  
  50.         Set<String> annotations = new LinkedHashSet<>();  
  51.         annotations.add(JPHelloWorld.class.getCanonicalName());  
  52.         return annotations;  
  53.     }  
  54.   
  55.     @Override  
  56.     public SourceVersion getSupportedSourceVersion() {  
  57.         return SourceVersion.latestSupported();  
  58.     }  
  59. }  

在用 JavaPoet 来生成一个HelloWorld.java文件之前,我们还必须在 init()方法里获取到 Filer,这是一个用来辅助创建文件的接口,我们生成文件都通过它来处理。在 process()方法先创建了个 MethodSpec 表示一个方法,再创建一个 TypeSpec 表示一个类并添加上前面创建的方法,最后用 JavaFile 来生成对应的HelloWorld.java 并写入文件。

这是最简单的例子,整个语法结构也很清晰,相信做编程的看到这些使用方法都能猜到是做什么用的,我就没详细说了。除了这个例子,Github上还有很多其它示例,如果你想很好地了解编译时注解的使用的话,还是很有必要把每个例子都过一遍,如果不想自己敲粘贴复制下很容易的。

在代码中使用定义的注解:

[java] view plain copy
 在CODE上查看代码片派生到我的代码片
  1. @JPHelloWorld  
  2. public class MainActivity extends AppCompatActivity{  
  3.     // ...  
  4. }  

重新 Make 下工程就可以看到生成的 HelloWorld.java 文件了,目录如下:


可以看到已经成功生成了 HelloWorld.java 文件,注意生成文件所在的目录,现在我们就可以在项目中直接使用这个 java 类了。当然了,这个例子没有什么实际的使用价值,你可以参考其它例子来生成你想要的代码,用法是很多的。

这里只写了个最简单的例子,没有深入更详细的使用方法,等后面有时间再来整理个更详细的介绍。

自定义注解之编译时注解(RetentionPolicy.CLASS)(三)—— 常用接口介绍

源码:AnnotationSample

0 0
原创粉丝点击