平时作项目中有个很是好用的一个插件,叫lombok.它提供了一些简单的注解,能够用来生成javabean和一些getter/setter方法,提升了开发的效率节省了开发时间.
今天咱们就来看看lombok使用的什么方式来实现这种操做的.其实lombok使用的是annotation processor,这个是jdk1.5中增长的新功能.像@Getter只是一个注解,它真正的处理部分
是在注解处理器里面实现的.官方参考连接.html
注解处理器其实全称叫Pluggable Annotation Processing API,插入式注解处理器,它是对JSR269提案的实现,具体能够看连接里面的内容,JSR269连接.
它是怎么工做的呢?能够参考下图:
1.parse and enter:解析和输入,java编译器这个阶段会把源代码解析生成AST(抽象语法分析树)
2.annotation processing:注解处理器阶段,此时将调用注解处理器,这时候能够校验代码,生成新文件等等(处理完能够循环到第一步)
3.analyse and generate:分析和生成,此时前两步完成后,生成字节码(这个阶段进行了解糖,好比类型擦除)
这些其实只是为了给你们留有一个粗浅的印象,它是怎么执行的.java
看了上面的资料,大脑中应该有了一个大概的印象,如今咱们实际操做一下写一个简单的例子,实践一下.
要使用注解处理器须要两个步骤:
1.自定义一个注解
2.继承AbstractProcessor而且实现process方法git
咱们接下来写一个很简单的例子,就是在一个类上加上@InterfaceAnnotation,编译的时候去生成一个"I"+类名的接口类.
首先我这里是定义了两个moudle,一个用来写注解和处理器,另外一个用来调用注解.github
@Target({ElementType.TYPE}) @Retention(RetentionPolicy.SOURCE) public @interface InterfaceAnnotation { }
1.@Target:表示的是这个注解在什么上面使用,这里ElementType.TYPE是指在类上使用该注解
2.@Retention:表示的是保留到什么阶段,这里RetentionPolicy.SOURCE是源代码阶段,编译后的class上就没有这个注解了apache
@SupportedAnnotationTypes(value = {"com.example.processor.InterfaceAnnotation"}) @SupportedSourceVersion(value = SourceVersion.RELEASE_8) public class InterfaceProcessor extends AbstractProcessor { @Override public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) { Messager messager = processingEnv.getMessager(); messager.printMessage(Diagnostic.Kind.NOTE, "进入到InterfaceProcessor中了~~~"); // 将带有InterfaceProcessor的类给找出来 Set<? extends Element> clazz = roundEnv.getElementsAnnotatedWith(InterfaceAnnotation.class); clazz.forEach(item -> { // 生成一个 I + 类名的接口类 String className = item.getSimpleName().toString(); className = "I" + className.substring(0, 1) + className.substring(1); TypeSpec typeSpec = TypeSpec.interfaceBuilder(className).addModifiers(Modifier.PUBLIC).build(); try { // 生成java文件 JavaFile.builder("com.example.processor", typeSpec).build().writeTo(new File("./src/main/java/")); } catch (IOException e) { e.printStackTrace(); } }); return true; } }
1.@SupportedAnnotationTypes:表示这个processor类要对什么注解生效
2.@SupportedSourceVersion:表示支持的java版本
3.annotations:被要求的注解,就是@SupportedAnnotationTypes对应的注解
4.roundEnv:存放着当前和上一轮processing的环境信息
5.TypeSpec这个可能有点没看懂是干吗的,它是javaPoet中的一个类,javaPoet是java用于生成java文件的一款第三方插件很好用,因此这里使用了这个类来生成java文件,
实际上这里用java自带的PrintWriter等输入输出流也能够生成java文件,生成文件有不少方式.javaPoet的连接.javaPoet使用指南.
6.Messager是用来打印输出信息的,System.out.println其实也能够;
7.process若是返回是true后续的注解处理器就不会再处理这个注解,若是是false,在下一轮processing中,其余注解处理器也会来处理改注解.api
写好以后,这里须要指定processor,META-INF/services/javax.annotation.processing.Processor 写好com.example.processor.InterfaceProcessor.若是你不知道这是啥,能够看下我另外一篇博客(实力推广XD)什么是SPI
咱们在把注解处理器给编译好,maven里插件的设置:oracle
<plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>maven-compiler-plugin</artifactId> <version>3.7.0</version> <configuration> <source>1.8</source> <target>1.8</target> <!-- 不加这一句编译会报找不到processor的异常--> <compilerArgument>-proc:none</compilerArgument> </configuration> </plugin>
此时的目录结构是这样:maven
. ├── HELP.md ├── pom.xml ├── processor.iml └── src └── main ├── java │ └── com │ └── example │ └── processor │ ├── InterfaceAnnotation.java │ └── InterfaceProcessor.java └── resources └── META-INF └── services └── javax.annotation.processing.Processor
而后mvn clean install.ide
在使用以前呢,注解处理器要是编译好的.引入注解处理器的jar包.
测试类加上@InterfaceAnnotationpost
@InterfaceAnnotation public class TestProcessor { }
maven指定编译时使用的注解处理器.
<plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>maven-compiler-plugin</artifactId> <version>3.7.0</version> <configuration> <source>1.8</source> <target>1.8</target> <encoding>UTF-8</encoding> <annotationProcessors> <annotationProcessor> com.example.processor.InterfaceProcessor </annotationProcessor> </annotationProcessors> </configuration> </plugin>
此时目录结构是
. ├── HELP.md ├── pom.xml ├── src │ └── main │ ├── java │ │ └── com │ │ └── example │ │ └── test │ │ └── TestProcessor.java │ └── resources └── test.iml
而后mvn compile,生成了java文件,此时目录结构是:
. ├── HELP.md ├── pom.xml ├── src │ └── main │ ├── java │ │ └── com │ │ └── example │ │ ├── processor │ │ │ └── ITestProcessor.java // 这里就是生成的java文件 │ │ └── test │ │ └── TestProcessor.java │ └── resources ├── target │ ├── classes │ │ └── com │ │ └── example │ │ └── test │ │ └── TestProcessor.class │ ├── generated-sources │ │ └── annotations │ └── maven-status │ └── maven-compiler-plugin │ └── compile │ └── default-compile │ ├── createdFiles.lst │ └── inputFiles.lst └── test.iml
看到了生成的java文件就大功告成~
1.java注解处理器在不少地方均可以使用,实际应用好比lombok,安卓生成fragment等等,只使用一个注解能够省去不少代码,提升效率;
2.本文只是列举了一个很简单的例子,不少注解处理器里面的api都没有使用到,读者有兴趣的能够自行研究,并且有涉及到抽象语法树的api;
3.注解处理器能够用于生成新的类来完成某些功能,可是不能直接修改当前的类.
1.https://docs.oracle.com/javas...
2.https://jcp.org/aboutJava/com...
3.https://github.com/square/jav...
4.https://www.cnblogs.com/throw...
5.http://notatube.blogspot.com/...
6.https://www.baeldung.com/java...
7.http://hannesdorfmann.com/ann...