關(guān)于 APT 以及遇到的一些問(wèn)題

APT

Annotation Processing Tool 注解處理工具

在編譯時(shí)尺碰,掃描處理注解信息

常用于生成代碼,如 Dagger , EventBus , Butter Knife , Data Binding 這些庫(kù)

KAPT

Kotlin Annotation Processing Tool Kotlin 注解處理工具

kapt 的用法詳見(jiàn) 官方參考文檔

實(shí)踐

參考 EventBusButter Knife

1. 新建 Module annotation 竟贯,類型為 Java Library

build.gradle
apply plugin: 'java-library'

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

sourceCompatibility = JavaVersion.VERSION_1_8
targetCompatibility = JavaVersion.VERSION_1_8
創(chuàng)建注解
@Retention(RetentionPolicy.CLASS) // 注解保留到字節(jié)碼文件,即編譯期
@Target(ElementType.TYPE)       // 可作用于類泄鹏、接口(包括注解)朱巨、枚舉上
public @interface Function {
    String name() default "";
}

2. 新建 Module compiler 叙甸,類型為 Java Library

build.gradle
apply plugin: 'java-library'

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

sourceCompatibility = JavaVersion.VERSION_1_8
targetCompatibility = JavaVersion.VERSION_1_8
創(chuàng)建注解處理器
public class FunctionAnnotationProcessor extends AbstractProcessor {

    private Messager mMessager;

    @Override
    public synchronized void init(ProcessingEnvironment processingEnvironment) {
        super.init(processingEnvironment);
        mMessager = processingEnvironment.getMessager();
    }

    @Override
    public boolean process(Set<? extends TypeElement> set, RoundEnvironment roundEnvironment) {
        Set<? extends Element> elements = roundEnvironment.getElementsAnnotatedWith(Function.class);
        for (Element element : elements) {
            mMessager.printMessage(Diagnostic.Kind.WARNING, "find element: " + element.getSimpleName());
        }
        return false;
    }

    @Override
    public Set<String> getSupportedAnnotationTypes() {
        Set<String> types = new LinkedHashSet<>();
        types.add(Function.class.getName());
        return types;
    }

    @Override
    public SourceVersion getSupportedSourceVersion() {
        return SourceVersion.latest();
    }
}
添加 SPI (Service Provider Interface) 配置文件
  • main 目錄下創(chuàng)建 resources/META-INF/services 目錄颖医,新建 javax.annotation.processing.Processor 文件
  • 在該文件中添加注解處理器的完整類名

    me.jack.compiler.FunctionAnnotationProcessor
    

3. 在 Application Module 中使用注解

build.gradle
    ...
dependencies {
    ...
    implementation project(':annotation')
    annotationProcessor project(':compiler')
}
添加注解到類上
@Function(name = "Home")
public class HomeActivity extends AppCompatActivity {
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_home);
    }
}
點(diǎn)擊 Make Project ,執(zhí)行編譯

可以看到 Build Output 出現(xiàn)以下輸出

擴(kuò)展

使用 Google 的 auto-service 庫(kù)簡(jiǎn)化 SPI 配置文件操作

compiler module 的 build.gradle 添加依賴
compileOnly 'com.google.auto.service:auto-service:1.0-rc4'
annotationProcessor 'com.google.auto.service:auto-service:1.0-rc4'
FunctionAnnotationProcessor 類添加注解
@AutoService(Processor.class)
public class FunctionAnnotationProcessor extends AbstractProcessor {
    ...
}
刪除 resources 目錄
執(zhí)行 Build -> Rebuild Project 操作

依然可以看到輸出

find element: HomeActivity

使用 Kotlin 編寫

需要在 build.gradle 中添加 Kotlin 依賴裆蒸,并使用 kapt 替換 annotationProcessor

Module compilerbuild.gradle 文件如下

apply plugin: 'java-library'
apply plugin: 'kotlin'
apply plugin: 'kotlin-kapt'

dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation"org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version"
    implementation project(':annotation')
    compileOnly 'com.google.auto.service:auto-service:1.0-rc4'
    kapt 'com.google.auto.service:auto-service:1.0-rc4'
}

sourceCompatibility = JavaVersion.VERSION_1_8
targetCompatibility = JavaVersion.VERSION_1_8
Application Module 如果使用注解的類是 Kotlin 編寫熔萧,則需要在 build.gradle 同樣使用 kapt 替換 annotationProcessor ,否則處理器掃描不到該類
    ...
dependencies {
    ...
    implementation project(':annotation')
    kapt project(':compiler')
}

遇到的一些問(wèn)題

1. 編譯警告 Incremental annotation processing requested

原因

這是 Kotlin 1.3.50 的 bug僚祷,自 1.3.31 起佛致,kapt 支持增量注解處理,詳見(jiàn)官方文檔增量編譯

解決方法
  • 版本降級(jí)

    使用 1.3.41 或以下版本

  • 禁用增量編譯

    在項(xiàng)目根目錄下的 gradle.properties 文件中辙谜,添加 kapt.incremental.apt=false

2. 編譯警告 unknown enum constant AnnotationTarget.XXX

原因

混編俺榆,編譯時(shí)找不到 Kotlin 的類

解決方法

添加 Kotlin 依賴

dependencies {
    ...
    implementation"org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version"
}

3. 提示 Kotlin not configured

原因

沒(méi)添加 Kotlin 依賴

解決方法

build.gradle 中添加依賴

dependencies {
    ...
    implementation"org.jetbrains.kotlin:kotlin-stdlib:$kotlin_version"
}

4. 編譯錯(cuò)誤 error: cannot find symbol class XXX

原因

沒(méi)使用 Kotlin 插件

解決方法

build.gradle 中添加插件

apply plugin: 'kotlin'

5. Build Output 沒(méi)有打印注解處理器的信息

原因

處理器使用 Kotlin 編寫,而 build.gradle 中使用 annotationProcessor

解決方法

使用 kapt 替換 annotationProcessor

6. 編譯錯(cuò)誤 Annotation processors must be explicitly declared now

原因

注解的定義和處理器在同一個(gè)module中

解決方法
  • 按照提示筷弦,可以在 build.gradle 中添加配置肋演,但這種操作已經(jīng)過(guò)期并且之后會(huì)移除,不建議使用

    android {
        ...
        defaultConfig {
            ...
            javaCompileOptions.annotationProcessorOptions.includeCompileClasspath = true
        }
    }
    
  • 分離 注解的定義 和 處理器

    實(shí)踐 小節(jié)中烂琴,注解的定義為 annotation module,而處理器為 compiler module

源碼查看

Github地址

參考鏈接

  1. Android注解&APT技術(shù)
  2. Kotlin編譯警告問(wèn)題
  3. Android模擬ButterKnife BindView注解實(shí)現(xiàn)綁定view
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末蜕乡,一起剝皮案震驚了整個(gè)濱河市奸绷,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌层玲,老刑警劉巖号醉,帶你破解...
    沈念sama閱讀 218,858評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異辛块,居然都是意外死亡畔派,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,372評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門润绵,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)线椰,“玉大人,你說(shuō)我怎么就攤上這事尘盼『┯洌” “怎么了?”我有些...
    開(kāi)封第一講書人閱讀 165,282評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵卿捎,是天一觀的道長(zhǎng)配紫。 經(jīng)常有香客問(wèn)我,道長(zhǎng)午阵,這世上最難降的妖魔是什么躺孝? 我笑而不...
    開(kāi)封第一講書人閱讀 58,842評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上植袍,老公的妹妹穿的比我還像新娘伪很。我一直安慰自己,他們只是感情好奋单,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,857評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布锉试。 她就那樣靜靜地躺著,像睡著了一般览濒。 火紅的嫁衣襯著肌膚如雪呆盖。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書人閱讀 51,679評(píng)論 1 305
  • 那天贷笛,我揣著相機(jī)與錄音应又,去河邊找鬼。 笑死乏苦,一個(gè)胖子當(dāng)著我的面吹牛株扛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播汇荐,決...
    沈念sama閱讀 40,406評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼洞就,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了掀淘?” 一聲冷哼從身側(cè)響起旬蟋,我...
    開(kāi)封第一講書人閱讀 39,311評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎革娄,沒(méi)想到半個(gè)月后倾贰,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,767評(píng)論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡拦惋,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,945評(píng)論 3 336
  • 正文 我和宋清朗相戀三年匆浙,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片厕妖。...
    茶點(diǎn)故事閱讀 40,090評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡首尼,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出叹放,到底是詐尸還是另有隱情饰恕,我是刑警寧澤,帶...
    沈念sama閱讀 35,785評(píng)論 5 346
  • 正文 年R本政府宣布井仰,位于F島的核電站埋嵌,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏俱恶。R本人自食惡果不足惜雹嗦,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,420評(píng)論 3 331
  • 文/蒙蒙 一范舀、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧了罪,春花似錦锭环、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 31,988評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至娃圆,卻和暖如春玫锋,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背讼呢。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 33,101評(píng)論 1 271
  • 我被黑心中介騙來(lái)泰國(guó)打工撩鹿, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人悦屏。 一個(gè)月前我還...
    沈念sama閱讀 48,298評(píng)論 3 372
  • 正文 我出身青樓节沦,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親础爬。 傳聞我的和親對(duì)象是個(gè)殘疾皇子甫贯,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,033評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容