APT的使用1

作者: ZenCabin | 来源:发表于2018-07-18 23:05 被阅读257次

APT(Annotation Processing Tool) 注解编译时工具。现在越来越多的框架使用apt技术来实现或重写,如Dagger2、ButterKnight、ARouter。APT技术可以简单理解为在编译时通过处理注解来实现部分代码逻辑,动态生成java class文件,供我们使用。

注解的介绍

APT系列:
APT的使用1(apt的接入)
APT的使用2(实现findViewById功能)


下面通过一个简单的apt接入实例来了解apt的基本用法

1. 新建一个名为lib的java library

lib的build.gradle

apply plugin: 'java-library'

dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
}

sourceCompatibility = "1.7"
targetCompatibility = "1.7"

添加一个注解类,如@Test

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.CLASS)
public @interface Test {
    String path();
}

2. 新建一个名为apt-lib的java library,来实现处理注解的逻辑

apt-lib的build.gradle

apply plugin: 'java-library'

dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation project(':lib')
    implementation 'com.google.auto.service:auto-service:1.0-rc4'
    implementation 'com.squareup:javapoet:1.11.1'
}

sourceCompatibility = "1.7"
targetCompatibility = "1.7"

其中com.google.auto.service:auto-service:1.0-rc4是google开源的用于注册Processor的工具库,com.squareup:javapoet:1.11.1是一个开源的生产class文件的库。

实现一个简单的处理注解的processor类

@AutoService(Processor.class)
@SupportedSourceVersion(SourceVersion.RELEASE_7)
@SupportedAnnotationTypes({"com.example.lib.Test"})
public class MyProcessor extends AbstractProcessor {
    @Override
    public boolean process(Set<? extends TypeElement> set, RoundEnvironment roundEnvironment) {
        // 生成一个main方法,带参数,并声称语句,传入参数
        MethodSpec main = MethodSpec.methodBuilder("main")
                .addModifiers(Modifier.PUBLIC, Modifier.STATIC)
                .addParameter(String[].class,"args")
                .addStatement("$T.out.println($L)",System.class, "args[0]")   
                .build();
        // 生成一个类、接口,添加一个方法给它
        TypeSpec typeSpec = TypeSpec.classBuilder("HelloWord")
                .addModifiers(Modifier.PUBLIC,
                Modifier.FINAL).addMethod(main).build();
        // 生成java文件,传入类、包名等信息
        JavaFile file = JavaFile.builder("com.jason.test", typeSpec).build();

        try {
            // 写到我们的环境中去
            file.writeTo(processingEnv.getFiler());
        } catch (IOException e) {
            e.printStackTrace();
        }
        return true;
    }
}

上述代码中的 $L,$T 代表不同的字面量类型,具体参考这里

3.在app中配置依赖

打开app的build.gradle文件配置如下:

dependencies {
    xxx...
    compile project(':lib')    // 引入lib
    annotationProcessor project(':apt-lib')  // 指定使用apt处理我们的apt-lib库
}

注意:上述操作在gradle版本3.0.1及以上,如果低于这个版本,在根build.gradle文件中添加

// 在根build.gradle文件添加
classpath 'com.neenbedankt.gradle.plugins:android-apt:1.8'

// 在app的build.gralde中添加
apply plugin: 'com.neenbedankt.android-apt'

然后点击rebuild,就会在目录(app/build/generated/source/apt/debug/包名/Test)下生成我们的HelloWord文件了,名字在MyProcessor中指定了,可以很灵活的指定。

4.在app中使用注解类@Test

在项目编译期,annotationProcessor project(':TestCompiler')这里会处理项目中的Processor类。最后也可以在项目中操作Processor里生成的HelloWorld类。

@Test(path = "main")
public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        HelloWord hw = new HelloWord();
        hw.main(new String[]{"我是一个processor"});
    }
}

#APT的运行方式、时机及注意点#

  • 根据sun官方的解释,APT是一个命令行工具,它对源代码文件进行检测并找出其中的RetentionPolicy为CLASS的annotation,然后使用annotation processor来处理,这个annotation processor使用一套反射api并支持JSR175规范。

  • APT在编译时自动查找左右继承自AbstractProcessor的类,然后调用他们的process方法,相当于在编译过程中执行代码。

  • 在Android Studio中找不到AbstractProcessor类是因为android.jar默认不包含javax的包,所以我们需要新建一个Module,指定为java library,然后在其中实现我们的processor。

  • @SupportedAnnotationTypes({"com.example.lib.Test"})通过这段代码指定要处理的注解类;@SupportedSourceVersion(SourceVersion.RELEASE_7)通过这段代码,指定编译的版本,这种通过注解指定编译版本和类型的方式是从Java1.7才有的。对于之前的版本是通过重写AbstractProcessor中的方法来指定的。

相关文章

网友评论

    本文标题:APT的使用1

    本文链接:https://www.haomeiwen.com/subject/whhxjttx.html