Annotation & Annotation process Tool

前面是一些名词的解释和说明,例子创建可以直接看后面的“创建过程”的章节。

本文主要参考网上各个大神的总结和自己使用过程的问题完成的,如果你发现有你的文章的内容,请通知我,我将你的文章放在引用目录里,谢谢。
如有错误和需要勘正的地方,请指教。

Annotation

注解是JDK5.0引入的新特性
JDK5.0 当时提供了三个注解: @Deprecated(废弃的、过时的)、@Override(重写、覆盖)、@SuppressWarnings(压缩警告)

  1. 注解相当于一种标记,在程序中加入注解就等于为程序打上某种标记。
  2. javac编译器、开发工具和其他程序可以通过反射来了解类及各元素上的标记信息。
  3. 注解可以加在包、类、属性、方法、方法的参数以及局部变量上。

元注解(meta-annotaion)

JDK5.0引入,负责注解其他注解

元注解包含:

  1. @Target:规定自定义Annotation所修饰的对象范围
    • ElementType.CONSTRUCTOR,构造器声明
    • ElementType.FIELD,成员变量、对象、属性(包括enum实例)
    • ElementType.LOCAL_VARIABLE,局部变量声明
    • ElementType.METHOD,方法声明
    • ElementType.PACKAGE,包声明
    • ElementType.PARAMETER,参数声明
    • ElementType.TYPE,类、接口(包括注解类型)或enum声明
    • 同时支持多种范围设定,e.g.:
      @Target({ElementType.TYPE,ElementType.FIELD})
  2. @Retention:对Annotation的“生命周期”限制,表示需要在什么级别保存该注释信息
    • RetentionPolicy.SOURCE,在源文件中有效
    • RetentionPolicy.CLASS,在class文件中有效
    • RetentionPolicy.RUNTIME,在运行时有效
  3. @Documented:用于描述其他类型的annotation应该被作为被标注的程序成员的公共API,因此可以被例如javadoc此类工具文档化,Documented是一个标记注解,没有成员
  4. @Inherited:标记注解,被标注的类型是被继承的,主要用于类。

自定义注解

使用@interface自定义注释时,自动继承了java.lang.annotation.Annotation接口,由编译程序自动完成其他细节。在定义注解时,不能继承其他的注解或接口。@interface用来声明一个注解,其中的每一个方法实际上是声明了一个配置参数。方法的名称就是参数的名称,返回值类型就是参数类型(只能是基本类型、Class、String、enum)。可以通过default来声明参数的默认值。

注解使用的参数

只能用基本类型byte,short,char,int,long,float,double,boolean八种基本数据类型和String,Enum,Class,Annotation等数据类型,以及这些类型的数组。
必须有明确的值,可以在定义注解的默认值中指定;或者 使用注解时指定,非基本类型的注解元素的值不可为null。

自定义注解的局限

  1. 不能修改已有的源文件,可以创建新的源文件

APT(Used in Android Studio)

APT(Annotation Processing Tool)是一种处理注释的工具,它对源码文件进行检测找出其中的Annotation,使用Annotation进行额外的处理。
Annotation处理器在处理Annotation时可以根据源文件中的Annotation生成额外的源文件和其他的文件(文件具体内容由Annotation处理器的编写者决定),APT还会编译生成的源文件和原来的源文件,将它们一起生成class文件。

创建过程

  1. 创建一个Java Library(比如叫做annotation),放置自定义的annotation和关联代码。

    @Target(ElementType.TYPE)
    @Retention(RetentionPolicy.CLASS)  
    public @interface Test {   }
    
  2. 配置该library的build.gradle

    该配置信息在android studio将自动生成

    apply plugin: 'java' 
    sourceCompatibility = 1.7  
    targetCompatibility = 1.7   
    dependencies { 
        compile fileTree(dir: 'libs', include: ['*.jar']) 
    }  
    
  3. 创建一个Java library(比如叫做compiler),放置自定义的Annotation Processor。

    AutoService是一个开源库提供的注册Processor的方法。

    @AutoService(Processor.class)
    public class TestProcessor extends AbstractProcessor {
        @Override
        public Set<String> getSupportedAnnotationTypes() {
            return Collections.singleton(Test.class.getCanonicalName());
        }
        @Override
        public boolean process(Set<? extends TypeElement> annotations, RoundEnvironment roundEnv) {
            return false;
        }
    }
    
  4. 配置build.gradle

    apply plugin: 'java'
    sourceCompatibility = 1.7 
    targetCompatibility = 1.7 
    dependencies {
        compile fileTree(dir: 'libs', include: ['*.jar'])
        compile 'com.google.auto.service:auto-service:1.0-rc2'
        compile 'com.squareup:javapoet:1.7.0'
        compile project(':annotation')
    }
    
  5. 配置Project的build.gradle

    加在buildscript的dependencies中

    classpath 'com.neenbedankt.gradle.plugins:android-apt:1.8'
    
  6. 配置app的build.gradle

    增加pluge

    apply plugin: 'com.neenbedankt.android-apt'
    

    dependencies中增加

    compile project(':annotation')
    apt project(':compiler')
    
  7. 新版本android studio 5和6的配置会报错,只要直接使用annotationProcessor引入依赖就好,即源码如下:

    dependencies中增加

    compile project(':annotation')
    annotationProcessor project(':compiler')
    

Debug自定义Annotaion Processor

编译过程:

  1. javac启动一个JVM运行processor;
  2. compiler编译代码,所以processor生成的代码可以被编译进apk;
  3. IDE启动JVM运行代码。
  1. 将需要debug的processor加入compiler中

    • 在setting中打开Enable annotation processing(setting[需要在File->Other Settings->Default Setting 或者 Android studio欢迎页面中的Configure->Preferences]->Build,Execution,Deployment->Compiler->Annotation Processors)
    • Processor FQ Name中增加一个自定义Processor的全路径
  2. 设置javac为debug模式

    • 在Annotation Processors的设置项上面的Java Compiler中

    • 设置Additional command line parameters中设置:

      -J-Xdebug -J-Xrunjdwp:transport=dt_socket,server=y,suspend=y,address=5005
      
  3. 增加一个远程编译选项并启动编译

    • 打开项目的Run/Debug Configurations
    • 在左边栏中,点击“+”增加一个Remote,确定。
  4. 设置项目gradle.properties增加:

    org.gradle.jvmargs=-Xmx1536m -agentlib:jdwp=transport=dt_socket,server=y,suspend=n,address=5005
    
  5. 点击调试按钮,打开远程调试功能(这个时候自定义processor中的断点显示生效)

  6. 重新编译工程,断点响应

注1:1、2步骤在实际中发现,不是必须要设置的,具体1、2步骤会有什么作用,还在研究。
注2:当遇到unable to open debugger port: connection refused的错误时,是因为:

  1. 当前的端口号被占用,你需要找到一个你电脑未被占用的端口号进行调试
  2. 工程引用的所有gradle.properties文件中,存在未加调试命令的文件

常用操作

  1. 获取VariableElement的类型的TypeElement:Types.asElement(VariableElement.asType()):
    • 注意:如果类型是基本类型,则返回值是null,需要通过Types.boxedClass((PrimitiveType) VariableElement.asType());获取到装箱类的类型,基本类型只有PrimitiveType,他是一个TypeMirror。
  2. TypeElement可以通过getEnclosingElement().toString()获取包名,getSimpleName().toString()获取类名
  3. 数组的处理:asType(),返回ArrayType,getComponentType()返回是哪种类型的数组
  4. List的处理:asType(),返回DeclaredType,getTypeArguments()获取是哪种类型的list。

相关类说明

AbstractProcessor

  • init:ProcessingEnvironment,参数ProcessingEvnironment提供很多有用的工具类。
  • process:Set<TypeElement> RoundEnvironment,RoundEvnironment可以用来查询出包含特定注解的被注解元素。
    1. 注意问题:
      • 可能你使用的源文件还未编译,会抛出异常MirroredTypeException
      • 可能存在process多次调用,因为处理之后有新的文件生成,所以需要注意内部数据的更新和清空问题。
  • getSupportedAnnotationTypes,指定这个注解处理器是注册给哪个注解的。
  • getSupportedSourceVersion,指定你使用的Java版本。

ProcessingEnvironment

  • getElementUtils:获取element的操作类,用于扫描源码文件,源码的每一个部分都是一个element,如图:

    package com.example;    // PackageElement
    
    public class Foo {        // TypeElement
    
    private int a;      // VariableElement
    private Foo other;  // VariableElement
    
    public Foo () {}    // ExecuteableElement
    
    public void setA (  // ExecuteableElement
                     int newA   // TypeElement
                     ) {}
    }
    

    Element可以是类、方法、变量等等。

  • getMessager: 获取Messager的操作类,用于报告错误、警告 以及 提示信息的途径。

Test

  1. testCompile 'com.google.testing.compile:compile-testing:0.8':但是因为编译出来的文件中存在当前工程没有的类,所以虽然测试用例过了,但是依然会报错。

使用缺点

  1. 额外生成一定量的类:编译时间加长、包变大
  2. 可读性比较差:对代码的理解需要一定的门槛

引用文章

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

推荐阅读更多精彩内容