自定义Android注解Part2:代码自动生成

上一期我们已经把butterknife-annotations中的注解变量都已经定义好了,分别为BindView、OnClick与Keep。

如果你是第一次进入本系列文章,强烈推荐跳到文章末尾查看上篇文章,要不然你可能会有点云里雾里。

如果在代码中引用的话,它将与开源库ButterKnife的操作类似。

class MainActivity : AppCompatActivity() {
 
    @BindView(R.id.public_service, R.string.public_service)
    lateinit var sName: TextView
 
    @BindView(R.id.personal_wx, R.string.personal_wx)
    lateinit var sPhone: TextView
 
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)
        Butterknife.bind(this)
    }
 
    @OnClick(R.id.public_service)
    fun nameClick(view: View) {
        Toast.makeText(this, getString(R.string.public_service_click_toast), Toast.LENGTH_LONG).show()
    }
 
    @OnClick(R.id.personal_wx)
    fun phoneClick(view: View) {
        Toast.makeText(this, getString(R.string.personal_wx_click_toast), Toast.LENGTH_LONG).show()
    }
}

使用@BindView来绑定我的View,使用@OnClick来绑定View的点击事件。使用Butterknife.bind来绑定该Class,主要是用来实例化自动生成的类。(该部分下篇文章将提及)

我们自己定义的绑定注解库已经完成了1/3,接下来我们将实现它的代码自动生成部分。这时就到了上期提到的第二个Module:butterknife-compiler。

NameUtils是一些常量的管理工具类。

final class NameUtils {
 
    static String getAutoGeneratorTypeName(String typeName) {
        return typeName + ConstantUtils.BINDING_BUTTERKNIFE_SUFFIX;
    }
 
    static class Package{
        static final String ANDROID_VIEW = "android.view";
    }
 
    static class Class {
        static final String CLASS_VIEW = "View";
        static final String CLASS_ON_CLICK_LISTENER = "OnClickListener";
    }
 
    static class Method{
        static final String BIND_VIEW = "bindView";
        static final String SET_ON_CLICK_LISTENER = "setOnClickListener";
        static final String ON_CLICK = "onClick";
    }
 
    static class Variable{
        static final String ANDROID_ACTIVITY = "activity";
    }
}

NameUitls包含了自动生成的类名称,包名,方法名,变量名。总之就是为了代码更健全,方便管理。

第二个类Processor是今天的重中之重。也是注解库代码自动生成的核心部分。由于注解的自动生成代码都是在注解进程中进行,所以这里它继承于AbstractProcessor,其中主要有三个方法需要实现。

  1. init:初始化必要的数据
  2. getSupportedAnnotationTypes:所支持的注解
  3. process:解析注解,编写自动生成代码

init

从简单到容易,先是init方法,我们直接看代码

    @Override
    public synchronized void init(ProcessingEnvironment processingEnv) {
        super.init(processingEnv);
        mFiler = processingEnv.getFiler();
        mMessager = processingEnv.getMessager();
        mElementUtils = processingEnv.getElementUtils();
    }

方法参数processingEnv为我们提供注解处理所需的环境状态。我们通过getFiler()、getMessager()与getElementUthis()方法,分别获取创建源代码的Filer、消息发送器Messager(主要用于向外界发送错误信息)与解析注解元素所需的通用方法。

例如:当我们已经构建好了需要自动生成的类,这时我们就可以使用Filter来将代码写入到java文件中,如遇错误使用Messager将错误信息发送出去。

//写入java文
try {
    JavaFile.builder(packageName, typeBuilder.build()).build().writeTo(mFiler)
} catch (IOException e) {
    mMessager.printMessage(Diagnostic.Kind.ERROR, e.getMessage(), typeElement);
}

代码中的JavaFile与typeBuilder都是JavaPoet中的类。JavaPote主要提供Java API来帮助生成.java资源文件。

getSupportedAnnotationTypes

    @Override
    public Set<String> getSupportedAnnotationTypes() {
        return new TreeSet<>(Arrays.asList(
                BindView.class.getCanonicalName(),
                OnClick.class.getCanonicalName(),
                Keep.class.getCanonicalName())
        );
    }

看方法名就知道了,包含所支持的注解,将其通过set集合来返回。这里将我们上一期自定义的注解添加到set集合中即可。

process

到了本篇文章的核心,process用来生成与注解相匹配的方法代码。通过解析Class中定义的注解,生成与注解相关联的类。

    @Override
    public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
        ....
        ....
        return true;
    }

提供了两个参数:annotations与roundEnv,分别代表需要处理的注解,这里就代表我们自定义的注解;注解处理器所需的环境,帮助进行解析注解。

在开始解析注解之前,我们应该先过滤我们所不需要的注解。回头看getSupportedAnnotationTypes方法,我们只支持BindView、OnClick与Keep这三个注解。为了解析出相匹配的注解,我们将这个逻辑单独抽离出来,交由getTypeElementsByAnnotationType来管理。

    private Set<TypeElement> getTypeElementsByAnnotationType(Set<? extends TypeElement> annotations, Set<? extends Element> elements) {
        Set<TypeElement> result = new HashSet<>();
        //遍历包含的 package class method
        for (Element element : elements) {
            //匹配 class or interface
            if (element instanceof TypeElement) {
                boolean found = false;
                //遍历class中包含的 filed method constructors
                for (Element subElement : element.getEnclosedElements()) {
                    //遍历element中包含的注释
                    for (AnnotationMirror annotationMirror : subElement.getAnnotationMirrors()) {
                        for (TypeElement annotation : annotations) {
                            //匹配注释
                            if (annotationMirror.getAnnotationType().asElement().equals(annotation)) {
                                result.add((TypeElement) element);
                                found = true;
                                break;
                            }
                        }
                        if (found) break;
                    }
                    if (found) break;
                }
            }
        }
        return result;
    }

首先理解Element是什么?Element代表程序中的包名、类、方法,这也是注解所支持的作用类型。然后再回到代码部分,已经给出详细代码注释。
该方法的作用就是获取到有我们自定义注解的class。这里介绍两个主要的方法

  1. getEnclosedElements():获元素中的闭包的注解元素,在我们的实例中元素为MainActivity(TypeElement,Type代表Class),而闭包的注解元素则为sName、sPhone、nameClick、phoneClick与onCreate。在这里简单的理解就是获取有注解的字段名、方法名
  2. getAnnotationMirrors():获取上述闭包元素的所有注解。这里分别为sName与sPhone上的@BindeView、nameClick与phoneClick上的@OnClick、onCreate上的@Override。

所以通过该方法最终返回的就是MainActivity,它将被转化为TypeElement类型返回,然后将由processing来处理。

我们再回到process方法中。通过getTypeElementsByAnnotationType()方法我们已经获取到了我们使用了自定义注解的TypeElement(MainActivity)。

//获取与annotation相匹配的TypeElement,即有注释声明的class
Set<TypeElement> elements = getTypeElementsByAnnotationType(annotations, roundEnv.getRootElements());

下面我们再获取构建类所需的相关信息。

//包名
String packageName = mElementUtils.getPackageOf(typeElement).getQualifiedName().toString();
//类名
String typeName = typeElement.getSimpleName().toString();
//全称类名
ClassName className = ClassName.get(packageName, typeName);
//自动生成类全称名
ClassName autoGenerationClassName = ClassName.get(packageName,
        NameUtils.getAutoGeneratorTypeName(typeName));
 
//构建自动生成的类
TypeSpec.Builder typeBuilder = TypeSpec.classBuilder(autoGenerationClassName)
        .addModifiers(Modifier.PUBLIC)
        .addAnnotation(Keep.class);

注释已经划分清楚了,可以分为四个步骤

  1. 获取对应typeElement的包名(这里获取的是com.idisfkj.androidapianalysis)
  2. 获取typeElement的SimpleName(这里为MainActivity字符串)
  3. 根据上述获取的包名与SimpleName来构建一个ClassName,为了后续声明方法的参数类型(这里为MainActivity类,注意是MainActivity类型)
  4. 构建需要自动生成的ClassName,这里使用NameUtils.getAutoGeneratorTypeName进行了统一命名(这里自动生成的类名为MainActivityBinding,都以原始类名后面加Binding)

所有信息准备完毕后,然后开始定义自动生成的类。这里通过使用TypeSpec.Builder来构建。它是JavaPoet中的类。

JavaPoet

由于直接使用JavaFileObject生成.java资源文件是非常麻烦的,所以推荐使用JavaPoet。JavaPoet是一个开源库,主要用来帮助方便快捷的生成.java的资源文件。想要全面了解的可以查看Github链接。为了帮助快速读懂该文章,这里对其中几个主要方法进行介绍。当然在使用前还需在butterknife-compiler中的builder.gradle添加依赖:

dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation project(':butterknife-annotations')
    implementation 'com.squareup:javapoet:1.11.1'
}

同时也将上一期我们自定义的注解Module引入。

  1. TypeSpec.Builder: 定义一个类
  2. addModifiers: 定义private、public与protected类型
  3. addAnnotation: 对Element元素添加注解。例如:@Keep
  4. TypeSpec.Builder -> addMethod: 添加方法
  5. MethodSpec -> addParameter: 为方法添加参数类型与参数名
  6. MethodSpec -> addStatement: 在方法中添加代码块。而其中的一些动态类型会使用占位符替代。例如:addStatement("N(N)", "bindView", "activity"),它将会生成bindView(activity)。占位符:N -> name,T -> type(ClassName), $L -> literals

有了上面的理解我们再来看下面的生成代码:

//构建自动生成的类
TypeSpec.Builder typeBuilder = TypeSpec.classBuilder(autoGenerationClassName)
        .addModifiers(Modifier.PUBLIC)
        .addAnnotation(Keep.class);
 
//添加构造方法
typeBuilder.addMethod(MethodSpec.constructorBuilder()
        .addModifiers(Modifier.PUBLIC)
        .addParameter(className, NameUtils.Variable.ANDROID_ACTIVITY)
        .addStatement("$N($N)",
                NameUtils.Method.BIND_VIEW,
                NameUtils.Variable.ANDROID_ACTIVITY)
        .addStatement("$N($N)",
                NameUtils.Method.SET_ON_CLICK_LISTENER,
                NameUtils.Variable.ANDROID_ACTIVITY)
        .build());

首先通过TypeSpec.Builder构建一个类,类名为autoGenerationClassName(MainActivity$Binding),类的访问级别为public,由于为了防止混淆使用了我们自定义的@Keep注解。

然后再来添加类的构造方法,使用addMethod、addModifiers、addParameter与addStatement分别构建构造方法名、方法访问级别、方法参数与方法中执行的代码块。所以上面的代码最终将会自动生成如下代码:

@Keep
public class MainActivity$Binding {
  public MainActivity$Binding(MainActivity activity) {
    bindView(activity);
    setOnClickListener(activity);
  }
}  

在自动生成类的构造方法中调用了我们想要的bindView与setOnClickListener方法。所以接下来我们要实现的就是这两个方法的构建。

bindView

//添加bindView成员方法
MethodSpec.Builder bindViewBuilder = MethodSpec.methodBuilder(NameUtils.Method.BIND_VIEW)
        .addModifiers(Modifier.PRIVATE)
        .returns(TypeName.VOID)
        .addParameter(className, NameUtils.Variable.ANDROID_ACTIVITY);
 
//添加方法内容
for (VariableElement variableElement : ElementFilter.fieldsIn(typeElement.getEnclosedElements())) {
    BindView bindView = variableElement.getAnnotation(BindView.class);
    if (bindView != null) {
        bindViewBuilder.addStatement("$N.$N=($T)$N.findViewById($L)",
                NameUtils.Variable.ANDROID_ACTIVITY,
                variableElement.getSimpleName(),
                variableElement,
                NameUtils.Variable.ANDROID_ACTIVITY,
                bindView.value()[0]
        ).addStatement("$N.$N.setText($N.getString($L))",
                NameUtils.Variable.ANDROID_ACTIVITY,
                variableElement.getSimpleName(),
                NameUtils.Variable.ANDROID_ACTIVITY,
                bindView.value()[1]);
    }
}
 
typeBuilder.addMethod(bindViewBuilder.build());

使用MethodSpec.Builder来创建bindView方法,其它的都与构造方法类似。使用returns为方法返回void类型。然后再遍历MainActivity中的注解,找到与我们定义的BindView相匹配的字段。最后分别向bindView方法中添加findViewById与setText代码块,同时将定义的方法添加到typeBuilder中。所以执行完上面代码后在MainActivity$Binding中展示如下:

  private void bindView(MainActivity activity) {
    activity.sName=(TextView)activity.findViewById(2131165265);
    activity.sName.setText(activity.getString(2131427362));
    activity.sPhone=(TextView)activity.findViewById(2131165262);
    activity.sPhone.setText(activity.getString(2131427360));
  }

实现了我们最初的View的绑定与TextView的默认值设置。

setOnClickListener

//添加setOnClickListener成员方法
MethodSpec.Builder setOnClickListenerBuilder = MethodSpec.methodBuilder(NameUtils.Method.SET_ON_CLICK_LISTENER)
        .addModifiers(Modifier.PRIVATE)
        .returns(TypeName.VOID)
        .addParameter(className, NameUtils.Variable.ANDROID_ACTIVITY, Modifier.FINAL);
 
//添加方法内容
ClassName viewClassName = ClassName.get(NameUtils.Package.ANDROID_VIEW, NameUtils.Class.CLASS_VIEW);
ClassName onClickListenerClassName = ClassName.get(NameUtils.Package.ANDROID_VIEW, NameUtils.Class.CLASS_VIEW, NameUtils.Class.CLASS_ON_CLICK_LISTENER);
 
for (ExecutableElement executableElement : ElementFilter.methodsIn(typeElement.getEnclosedElements())) {
    OnClick onClick = executableElement.getAnnotation(OnClick.class);
    if (onClick != null) {
        //构建匿名class
        TypeSpec typeSpec = TypeSpec.anonymousClassBuilder("")
                .addSuperinterface(onClickListenerClassName)
                .addMethod(MethodSpec.methodBuilder(NameUtils.Method.ON_CLICK)
                        .addModifiers(Modifier.PUBLIC)
                        .addParameter(viewClassName, NameUtils.Class.CLASS_VIEW)
                        .returns(TypeName.VOID)
                        .addStatement("$N.$N($N)",
                                NameUtils.Variable.ANDROID_ACTIVITY,
                                executableElement.getSimpleName(),
                                NameUtils.Class.CLASS_VIEW)
                        .build())
                .build();

        setOnClickListenerBuilder.addStatement("$N.findViewById($L).setOnClickListener($L)",
                NameUtils.Variable.ANDROID_ACTIVITY,
                onClick.value(),
                typeSpec);
    }
}
 
typeBuilder.addMethod(setOnClickListenerBuilder.build());

与bindView方法不同的是,由于使用到了匿名类OnClickListener与类View,所以我们这里也要定义他们的ClassName,然后使用TypeSpec来生成匿名类。生成之后再添加到setOnClickListener方法中。最后再将setOnClickListener方法添加到MainActivity$Binding中。所以最终展示如下:

  private void setOnClickListener(final MainActivity activity) {
    activity.findViewById(2131165265).setOnClickListener(new View.OnClickListener() {
      public void onClick(View View) {
        activity.nameClick(View);
      }
    });
    activity.findViewById(2131165262).setOnClickListener(new View.OnClickListener() {
      public void onClick(View View) {
        activity.phoneClick(View);
      }
    });
  }

我们的MainActivity$Binding类就已经定义完成,最后再写入到java文件中

//写入java文件
try {
    JavaFile.builder(packageName, typeBuilder.build()).build().writeTo(mFiler);
} catch (IOException e) {
    mMessager.printMessage(Diagnostic.Kind.ERROR, e.getMessage(), typeElement);
}

services

在butterknife-compiler中,我们还需创建一个特定的目录:
butterknife-compiler/src/main/resources/META-INF/services

在services目录中,我们还需创建一个文件:javax.annotation.processing.Processor,该文件是用来告诉编译器,当它在编译代码的过程中正处于注解处理中时,会告诉注解处理器来自动生成哪些类。

所以我们在文件中将添加我们自定义的Processor路径

com.idisfkj.butterknife.compiler.Processor

这样注解器就会调用该指定的Processor。到这里整个butterknife-compiler就完成了,现在我们可以Make Project一下工程,完成之后就可以全局搜索到MainActivity$Binding文件了。或者在如下路径中查看:

/app/build/generated/source/kapt/debug/com/idisfkj/androidapianalysis/MainActivity$Binding.java

文章中的代码都可以在Github中获取到。使用时请将分支切换到feat_annotation_processing

相关文章

自定义Android注解Part1:注解变量

自定义Android注解Part3:绑定

关注

怪谈时间到了
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,755评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,369评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,799评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,910评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,096评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,159评论 3 411
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,917评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,360评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,673评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,814评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,509评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,156评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,123评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,641评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,728评论 2 351

推荐阅读更多精彩内容