注解处理器使用

本文同步至:http://blog.edreamoon.com/

注解的生命周期

注解生命周期即表明注解在代码的什么阶段生效,通过@Retention来指定,其值可以为以下三种:

  • SOURCE,源码注解, 注解只在源码中存在,javac在编译成class时会把Java源程序上的源码注解给去掉编译成.class文件就将相应的注解去掉。
  • CLASS,编译时注解,注解在源码、.class文件里面存在。
  • RUNTIME,运行时注解,在运行阶段还起作用

三个阶段简单表示为:java源文件–>class文件–>内存中的字节码

注解处理器(Annotation Processor)

注解处理器用来在编译时扫描和处理注解,是javac的一个工具。一般使用注解处理器在编译时生成新的java文件,新生成的java文件会在编译期同其它java文件一样被编译。
目前比较流行的一些库都用到了注解处理器,如dagger2、butterknife、Android提供的data binding,相对于使用反射极大的提高了效率。

案例

Talk is cheap, show me the code! 下面使用一个demo来说明如何使用注解处理器。这个Demo比较简单,只是在编译项目时,打印注解信息。

工程目录如下:

752773433.png

mj-annotate、mj-processor是java library,分别用于自定义注解、自定义注解处理器。

  1. 首先定义注解,这个很简单
@Retention(RetentionPolicy.SOURCE)
@Target({ElementType.FIELD, ElementType.METHOD, ElementType.TYPE})
public @interface BindTest {
    int value();
}

这里指定为源码注解,可以用到字段、方法、类上

  1. 在项目的Activity中使用此注解,在类、方法、字段中均使用了下。
@BindTest(R.layout.activity_main)
public class MainActivity extends AppCompatActivity {
 
    @BindTest(R.id.bt_activity_main)
    private Button button;
 
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
 
    @BindTest(R.id.bt_activity_main)
    public void onClickButton() {
    }
}

3.注解已经使用了,但这些注解并没有起到什么作用。如何让项目编译时处理这些注解并打印注解相关信息哪?接下来就是注解处理器显神威的时候了。
定义一个处理器需要继承AbstractProcessor,并覆盖其中的一些方法的实现。

public class MProcessor extends AbstractProcessor {
    @Override
    public boolean process(Set<? extends TypeElement> set, RoundEnvironment roundEnvironment) {
        /**
         * 处理注解的地方,注解的上面说到的生成Java文件就是在此处理的,这里可以获取注解相关的内容
         */
 
        Messager messager = processingEnv.getMessager();
        //遍历用BindView注解的元素
        for (Element element : roundEnvironment.getElementsAnnotatedWith(BindTest.class)) {
            BindView annotation = element.getAnnotation(BindView.class);
 
            int value = annotation.value(); //注解的值
            ElementKind kind = element.getKind();//注解作用的位置
 
            if (kind == ElementKind.FIELD) {//如果这个元素是一个方法
                VariableElement variableElement = (VariableElement) element;//转换成方法对应的element
                messager.printMessage(Diagnostic.Kind.NOTE, "field");
            } else if (kind == ElementKind.METHOD) {//如果这个元素是一个方法
                ExecutableElement executableElement = (ExecutableElement) element;//转成方法对应的element
                messager.printMessage(Diagnostic.Kind.NOTE, "method return type : " + executableElement.getReturnType().toString());
                List<? extends VariableElement> params = executableElement.getParameters();//获取方法所有的参数
            } else if (kind == ElementKind.CLASS) {
                TypeElement typeElement = (TypeElement) element;
                messager.printMessage(Diagnostic.Kind.NOTE, "class " + typeElement.getQualifiedName());
            }
 
            //打印注解相关的信息
            messager.printMessage(Diagnostic.Kind.NOTE, value + " : " + kind.name());
        }
 
        /**
         * 返回true表示该注解已经被处理, 后续不会再有其他处理器处理; 返回false表示仍可被其他处理器处理.
         */
        return false;
    }
 
    @Override
    public Set<String> getSupportedAnnotationTypes() {
        /**
         * 返回支持的注解类型,限定注解处理器用到哪些注解上
         */
        Set<String> types = new LinkedHashSet<>();
        types.add(BindTest.class.getCanonicalName());
        return types;
    }
 
    @Override
    public SourceVersion getSupportedSourceVersion() {
        /**
         *指定使用的Java版本,为了更好的兼容使用SourceVersion.latestSupported()
         */
        return SourceVersion.latestSupported();
    }
}

主要实现就是getSupportedAnnotationTypes、getSupportedSourceVersion、process,代码中已经解释很清楚了。其中针对AbstractProcessor,也可以使用@SupportedAnnotationTypes、@SupportedSourceVersion注解来代替getSupportedAnnotationTypes() 、getSupportedSourceVersion()。也就是代码简化为如下:

@SupportedAnnotationTypes({"com.mj.annotate.BindView"})
@SupportedSourceVersion(SourceVersion.RELEASE_7)
public class MProcessor extends AbstractProcessor {
    @Override
    public boolean process(Set<? extends TypeElement> set, RoundEnvironment roundEnvironment) {
        //..............省略..........
    }
}

但考虑到兼容性和可维护性,最好还是用覆盖对应方法的形式实现比较好。

处理器已经写好了,然后在 app module 中添加对 mj-processor 的依赖。

dependencies {
    compile fileTree(include: ['*.jar'], dir: 'libs')
    compile 'com.android.support:appcompat-v7:25.1.0'
    compile project(path: ':mj-annotate')
    compile project(path: ':mj-processor')
}

此时Rebuild下项目,并不会打印process中的log信息,还需要创建处理器说明文件: main->resources->META-INF/services->javax.annotation.processing.Processor,然后在文件中输入定义的处理器:com.mj.processor.MProcessor。如果有多个Processor,以换行切换。这个步骤要创建多级目录比较麻烦,我们可以使用AutoService注解简化,在mj-processor 中添加相应依赖即可使用:

dependencies {
    compile fileTree(include: ['*.jar'], dir: 'libs')
    compile project(path: ':mj-annotate')
    compile 'com.google.auto.service:auto-service:1.0-rc2'
}

在 MProcessor 类使用@AutoService(Processor.class),编译成jar包时即会自动生成上述文件。

此时再回过头看文章开头的目录结构,就和各步骤对应了。
此时再Rebuild下项目就会打印log:

519130143.png

注意,此处是log是在gradle console中,为了能看到log,最好是Rebuild,或者是clean下项目。

Processor运行环境

既然Processor是编译时处理各种注解的,而Processor也是java文件,那它是如何运行工作的哪?
Processor也是运行在虚拟机JVM中的,只不过是一个独立的JVM,javac会启动一个完整Java虚拟机来保证Processor的工作,所以在这里保证了java的运行环境。

APT

Processor是在编译期工作的,而我们的apk运行时就不再需要了,但反编译上面生成的apk会发现mj-processor的代码。

APT(Annotation Processing Tool)就很好解决了这个问题,它能在编译时期去依赖注解处理器并进行工作,但在生成 APK 时不会包含任何遗留的东西
修改工程目录下的 build.gradle,添加maven仓库、apt依赖

 repositories {
    jcenter()
    mavenCentral() //for apt
}
dependencies {
    classpath 'com.android.tools.build:gradle:2.2.3'
    classpath 'com.neenbedankt.gradle.plugins:android-apt:1.8' //for apt
}

然后在 app module 修改依赖:

dependencies {
    compile fileTree(include: ['*.jar'], dir: 'libs')
    compile 'com.android.support:appcompat-v7:25.1.0'
    compile project(':mj-annotate')
//    compile project(':mj-processor')
    apt project(':mj-processor') // 使用apt依赖
}

这样就可以去除apk中processor相关内容

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

推荐阅读更多精彩内容