APT,就是Annotation Processing Tool 的简称,简单来讲就是经过编码来动态获得解析Annotation的工具。通常分为两类:
1.运行时注解:好比大名鼎鼎的retrofit就是用运行时注解,经过动态代理来生成网络请求
2.编译时注解:好比Dagger2, ButterKnife, EventBus3java
这里咱们要实现一个怎样的功能呢?第一个就是给咱们的activity添加一个@Flag,而后当咱们编译的时候就会生成一个java main函数。第二个就是我简易版的butterknife。2个注解都是写在用一个插件中。好了下面直接开始。bash
首先咱们建立一个my_annotation的java module,这个项目只放咱们的注解文件,不涉及到注解处理等其余逻辑,关于注解处理咱们会新建一个module来处理。项目结构以下:
网络
首先咱们建立一个my_compiler的java module。项目结构以下:
ide
auto-service:这是google推出的方便咱们编写annotation插件,在没有这个这个库以前,咱们须要对咱们的插件作不少的配置才能使用,有了这个库之后就方便多了,下文会看到怎么用,这里就不介绍了。
javapoet:这是方便咱们在编译时动态生成class文件的,下文也会有具体怎么使用,这里不作过多解释。函数
关于上面2个库你们感兴趣能够去查找相关资料进行进一步了解。下面咱们看下真正的注解处理类:工具
/**
* @author Jin
*/
//来自auto-service 只要添加这个注解之后就不须要作其余配置,如今已经能够在项目中直接使用了
@AutoService(Processor.class)
public class FlagAnnotationProcessor extends AbstractProcessor {
/**
* getSupportedSourceVersion()方法返回 Java 版本 默认为Java6
*
* @return Java 版本
*/
@Override
public SourceVersion getSupportedSourceVersion() {
return SourceVersion.latestSupported();
}
/**
* 返回要处理的注解的结合 这里只处理RouterAnnotion类型的注解
*
* @return
*/
@Override
public Set<String> getSupportedAnnotationTypes() {
LinkedHashSet<String> types = new LinkedHashSet<>();
types.add(Flag.class.getCanonicalName());
return types;
}
/**
* 注解的具体处理类
*
* @param annotations
* @param roundEnv
* @return
*/
@Override
public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
//来自javapoet 动态生成方法
MethodSpec main = MethodSpec.methodBuilder("main")
.addModifiers(Modifier.PUBLIC, Modifier.STATIC)
.returns(void.class)
.addParameter(String[].class, "args")
.addStatement("$T.out.println($S)", System.class, "Hello, JavaPoet!")
.build();
//来自javapoet 动态生成类
TypeSpec helloWorld = TypeSpec.classBuilder("HelloWorld")
.addModifiers(Modifier.PUBLIC, Modifier.FINAL)
.addMethod(main)
.build();
//来自javapoet 动态生成文件
JavaFile javaFile = JavaFile.builder("com.jin.helloworld", helloWorld)
.build();
try {
javaFile.writeTo(processingEnv.getFiler());
} catch (IOException e) {
e.printStackTrace();
}
return true;
}
}复制代码
注解处理类已经写好了,下面咱们看下怎么关联到咱们的项目中。gradle
下面进入咱们的另外一个demo,简易版butterknife。ui
下面我直接添上注解和处理代码(相关解释会在注释中):this
@AutoService(Processor.class)
public class DIAnnotationProcessor extends AbstractProcessor {
private Filer mFiler;
private Elements elementUtils;
/**
* init()方法能够初始化拿到一些使用的工具,
* 好比文件相关的辅助类 Filer;元素相关的辅助类Elements;日志相关的辅助类Messager;
*
* @param processingEnv
*/
@Override
public synchronized void init(ProcessingEnvironment processingEnv) {
super.init(processingEnv);
mFiler = processingEnv.getFiler();
elementUtils = processingEnv.getElementUtils();
}
/**
* getSupportedSourceVersion()方法返回 Java 版本 默认为Java6
*
* @return Java 版本
*/
@Override
public SourceVersion getSupportedSourceVersion() {
return SourceVersion.latestSupported();
}
/**
* 返回要处理的注解的结合 这里只处理RouterAnnotion类型的注解
*
* @return
*/
@Override
public Set<String> getSupportedAnnotationTypes() {
LinkedHashSet<String> types = new LinkedHashSet<>();
types.add(BindActivity.class.getCanonicalName());
return types;
}
/**
* 注解的具体处理类
*
* @param annotations
* @param roundEnv
* @return
*/
@Override
public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
System.out.println("DIAnnotationProcessor");
//获得全部被Bind添加注解的类
Set<? extends Element> elements = roundEnv.getElementsAnnotatedWith(BindActivity.class);
for (Element element : elements) {
//强制转换成TypeElement 判断是不是Class
TypeElement typeElement = (TypeElement) element;
//获得typeElement类中全部成员变量和成员方法
List<? extends Element> members = elementUtils.getAllMembers(typeElement);
MethodSpec.Builder bindViewMethodSpecBuilder = MethodSpec.methodBuilder("bindView")
.addModifiers(Modifier.PUBLIC, Modifier.STATIC)
.returns(TypeName.VOID)
.addParameter(ClassName.get(typeElement.asType()), "activity");
for (Element item : members) {
BindMyView bindView = item.getAnnotation(BindMyView.class);
if (bindView == null) {
continue;
}
bindViewMethodSpecBuilder.addStatement(String.format("activity.%s = (%s) activity.findViewById(%s)", item.getSimpleName(), ClassName.get(item.asType()).toString(), bindView.value()));
}
TypeSpec typeSpec = TypeSpec.classBuilder("DI" + element.getSimpleName())
.superclass(TypeName.get(typeElement.asType()))
.addModifiers(Modifier.PUBLIC, Modifier.FINAL)
.addMethod(bindViewMethodSpecBuilder.build())
.build();
JavaFile javaFile = JavaFile.builder(getPackageName(typeElement), typeSpec).build();
try {
javaFile.writeTo(processingEnv.getFiler());
} catch (IOException e) {
e.printStackTrace();
}
}
return true;
}
private String getPackageName(TypeElement type) {
return elementUtils.getPackageOf(type).getQualifiedName().toString();
}
}复制代码
#调试(AndroidStudio3.0)已解决
若是过你也像我同样按照网上的教程操做,可是始终报错:Unable to open debugger port (localhost:5006): java.net.ConnectException "Connection refused: connect"
配置以下:
google