Java注解知多少之自定义注解处理器

Java注解知多少之自定义注解处理器

前言

要知道如何自定义注解处理器(简称:APT),首先还是先得知道注解与注解处理器是怎么一会儿事儿,关于自定义注解,可以看我之前的文章,现在主要讲的是注解处理器的使用。

简单来说注解处理器其实就是用来处理注解用的,在编译的运行成class文件的时候,注解处理器会根据需要处理的注解执行一些操作。

下面我们来自定义个跟ButterKnife类似的注解处理器来了解他的大致用法。

一、自定义注解

https://www.jianshu.com/p/ff234438a87a

既然是要自定义注解处理器,自然少不了注解啦,首先创建一个java Module(butterknife_annotation

/**
 * @author Gentle Wen
 */

@Retention(RetentionPolicy.RUNTIME)//运行时注解
@Target(ElementType.FIELD)//作用域在成员变量
public @interface LsBindView {
    int value();//注解需要填入的值,就是你的view id
}

二、自定义注解处理器

下面就是处理注解的APT,首先我们创建一个java Module 并先引用一些依赖包,当然包括上面创建的注解的module

dependencies {
    //注解处理器需要用到的依赖,这里应用谷歌的
    implementation 'com.google.auto.service:auto-service:1.0-rc6'
    implementation project(':butterknife_annotation')
    annotationProcessor 'com.google.auto.service:auto-service:1.0-rc6'
}
/**
 * 创建日期:2020/5/30 0007
 *
 * @author :Gentle Wen
 * describe:
 */
@AutoService(Processor.class)
public class ButterKnifeProcessor extends AbstractProcessor {
    /**
     * 注解处理器元素工具类
     */
    private Elements elementUtils;
    /**
     * 注解处理器操作相关的
     */
    private Filer filer;
    /**
     * 注解处理器打印相关
     */
    private Messager messager;
//    private Map<String, JavaCodeCreator> javaCodeCreatorMap = new HashMap<>();

    /**
     * 注解处理器支持的注解
     *
     * @return
     */
    @Override
    public Set<String> getSupportedAnnotationTypes() {
        Set<String> set = new LinkedHashSet<>();
        set.add(LsBindView.class.getCanonicalName());
        return set;
    }

    /**
     * jdk版本号
     *
     * @return
     */
    @Override
    public SourceVersion getSupportedSourceVersion() {
        return SourceVersion.latestSupported();
    }

    /**
     * 初始化注解处理器的时候调用
     *
     * @param processingEnvironment
     */
    @Override
    public synchronized void init(ProcessingEnvironment processingEnvironment) {
        super.init(processingEnvironment);
        //元素的工具
        elementUtils = processingEnvironment.getElementUtils();
        //用code String生成一个java文件调用的类对象
        filer = processingEnvironment.getFiler();
        //打印
        messager = processingEnvironment.getMessager();

    }

    /**
     * 注解处理器的入口
     *
     * @param set
     * @param roundEnvironment
     * @return
     */
    @Override
    public boolean process(Set<? extends TypeElement> set, RoundEnvironment roundEnvironment) {
        messager.printMessage(Diagnostic.Kind.NOTE,"process start....");
        try {
            JavaCodeCreator javaCodeCreator = null;
            //得到有这个LsBindView.class所有元素
            Set<? extends Element> elements = roundEnvironment.getElementsAnnotatedWith(LsBindView.class);
            for (Element element : elements) {
                //成员变量的元素
                VariableElement variableElement = (VariableElement) element;
                //Class节点
                TypeElement typeElement = (TypeElement) variableElement.getEnclosingElement();
                //注解
                LsBindView annotation = variableElement.getAnnotation(LsBindView.class);
                //viewId
                int viewId = annotation.value();
                if (javaCodeCreator == null) {
                    //创建一个java代码生成的工具类
                    javaCodeCreator = new JavaCodeCreator(typeElement, elementUtils);
                }
                javaCodeCreator.saveVariableElement(viewId, variableElement);
            }
            if (javaCodeCreator != null) {
                //在工具类中拼接完我们需要的代码
                String code = javaCodeCreator.createCode();
                //打印
                messager.printMessage(Diagnostic.Kind.NOTE,code);
                //利用该对象创建出我们需要的java文件
                JavaFileObject sourceFile = filer.createSourceFile(javaCodeCreator.getFullClassName(), javaCodeCreator.getTypeElement());
                Writer writer = sourceFile.openWriter();
                writer.write(code);
                writer.flush();
                writer.close();
            }
        } catch (Exception e) {
            e.printStackTrace();
            messager.printMessage(Diagnostic.Kind.ERROR,e.getMessage());
        }
        messager.printMessage(Diagnostic.Kind.NOTE,"process finish....");
        return false;
    }
}

我们知道,使用ButterKnife会帮我们生成我们需要的辅助类的java文件,那么我们就用上面的注解处理器帮我们生成一个java文件,现在自定义个工具类(JavaCodeCreator),让代码帮我们生成指定格式的代码。

/**
 * 创建日期:2020/5/30 0007
 *
 * @author :Gentle Wen
 * describe:
 */
 
public class JavaCodeCreator {
    private Map<Integer, VariableElement> variableElements = new HashMap<>();
    private TypeElement typeElement;

    private String packageName;

    public JavaCodeCreator(TypeElement typeElement, Elements elementUtils) {
        this.typeElement = typeElement;
        this.packageName = elementUtils.getPackageOf(typeElement).getQualifiedName().toString();
    }

    /**
     * 创建代码
     *
     * @return
     */
    public String createCode() {
        StringBuilder stringBuilder = new StringBuilder();
        //创建package com.xx.xxx;
        stringBuilder.append("package ").append(packageName).append(";").append("\n");
         //创建import com.xx.xx.XxActivity;
        stringBuilder.append("import ").append(typeElement.getQualifiedName().toString()).append(";").append("\n");
        //创建public class XxActivity_LsViewBinding{还有方法}
        stringBuilder.append("public class ").append(typeElement.getSimpleName()).append("_LsViewBinding").append("{\n");
        stringBuilder.append(createMethod());
        stringBuilder.append("}\n");

        return stringBuilder.toString();
    }

    /**
     * 创建bind方法
     * 这里帮我们创建出我们需要的方法
     * @return
     */
    private String createMethod() {
        String activityStr = "activity";
        String findViewById = "findViewById";
        StringBuilder stringBuilder = new StringBuilder();
        //public void bind(com.xxx.xxx.XxActivity activity){
        stringBuilder.append("\t\tpublic void bind").append("(").append(typeElement.getQualifiedName()).append(" ").append(activityStr).append(")").append("{\n");

        for (Integer integer : variableElements.keySet()) {
            VariableElement variableElement = variableElements.get(integer);
            stringBuilder.append("\t\t\t\t").append(activityStr)
                    .append(".").append(variableElement.getSimpleName()).append(" = ")
                    .append("(").append(variableElement.asType().toString()).append(")")
                    .append(activityStr).append(".").append(findViewById).append("(").append(integer).append(")").append(";\n");

        }
        stringBuilder.append("\t\t}\n");
        return stringBuilder.toString();
    }

    public void saveVariableElement(int viewId, VariableElement variableElement) {
        variableElements.put(viewId, variableElement);
    }

    public String getFullClassName() {
        return typeElement.getQualifiedName().toString()+"_LsViewBinding";
    }

    public Element getTypeElement() {
        return typeElement;
    }
}

三、工具类

创建一个java Module (butterknife_utils),用于传入我们的activity,调用我们生成的XxActivity_LsViewBinding对象的方法

/**
 * 创建日期:2020/5/30  0007
 *
 * @author :Gentle Wen
 * describe:
 */
public class ButterKnifeManager {
    public static void inject(Activity activity){
        try {
            //你当前activity的对象的class
            Class activityClass = activity.getClass();
            //根据你当前对象的路径反射找到我们生成的XxActivity_LsViewBinding路径
            Class clazz = Class.forName(activityClass.getName()+"_LsViewBinding");
            //反射我们的bindView方法,形参就是我们的activity
            Method method = clazz.getMethod("bind", activityClass);
            //带入activity参数执行我们的bindView方法
            method.invoke(clazz.newInstance(),activity);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

四、项目中使用

1.build.gradle

defaultConfig {
    javaCompileOptions { annotationProcessorOptions { includeCompileClasspath = true } }
}   
implementation project(':butterknife_utils')
implementation project(':butterknife_annotation')
annotationProcessor project(':butterknife')

2.MainActivity

package com.xx.xx;

import android.app.Activity;
import android.os.Bundle;
import android.util.Log;
import android.widget.Button;

import com.xx.butterknife_annotation.LsBindView;
import com.xx.butterknife_utils.ButterKnifeManager;

public class MainActivity extends Activity {
    @LsBindView(R.id.button)
    Button button;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ButterKnifeManager.inject(this);
        button.setOnClickListener(v -> {
            Log.i("MainActivity", "onClick: " + "LsBindView实现");
        });
    }

}

3.build一下你的project,之后就会在你的build文件夹路径下生成一个MainActivity_LsViewBinding的java文件

package com.xx.xx;
import com.xx.xx.MainActivity;
public class MainActivity_LsViewBinding{
      public void bind(com.xx.xx.MainActivity activity){
            activity.button = (android.widget.Button)activity.findViewById(2130968578);
      }
}

4.运行你的项目,点击按钮

MainActivity: onClick: LsBindView实现

总结

综上所述,我们可以知道,自定义注解处理器可以帮我们实现我们简单却又不得不实现的逻辑。在众多的框架中我们都可以看到注解处理器的影子(如:ButterKnife、Daggar等),能够掌握APT,也能在往Android架构师的路上添砖加瓦啊。

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