這是一本從基礎(chǔ)知識、設(shè)計(jì)思想、技術(shù)方案、應(yīng)用方法、實(shí)踐技巧5個維度系統(tǒng)講解Kotlin元編程,并以此大幅提升Kotlin工程師開發(fā)水平、研發(fā)效率和開發(fā)體驗(yàn)的著作。作者是Kotlin領(lǐng)域的資深專家和布道者,本書源于他對Kotlin編譯器源碼的反復(fù)研讀和大量的工程實(shí)踐,不僅細(xì)致講解了反射、程序靜態(tài)分析、 Java注解處理器、Kotlin符號處理器、Kotlin編譯器插件、元程序的開發(fā)和調(diào)試等核心元編程技術(shù),而且詳細(xì)剖析了Jetpack Compose的編譯器插件和IntelliJ 插件、AtomicFU 的 JVM 字節(jié)碼和JavaScript代碼的生成邏輯。本書的出版打破了元編程技術(shù)資料少、門檻高的行業(yè)現(xiàn)狀。本書包含大量案例,這些案例大多來自真實(shí)的生產(chǎn)實(shí)踐,相對成熟和完善,可以作為元編程項(xiàng)目的范本。同時,本書提供大量的代碼,為了提升閱讀體驗(yàn),在注釋、書寫和排版等方面對代碼做了精心的優(yōu)化。全書的源文件均可免費(fèi)下載,讀者可以通過作者的網(wǎng)站實(shí)時與作者互動和交流。
(1)作者背景資深:作者先后就職于騰訊和猿輔導(dǎo),是中國Kotlin社區(qū)知名布道者和技術(shù)專家,Google開發(fā)者專家(Kotlin方向)(2)作者經(jīng)驗(yàn)豐富:作者在Kotlin領(lǐng)域有大量的項(xiàng)目實(shí)踐經(jīng)驗(yàn),對Kotlin編譯器源碼有深入研究,著有暢銷書《深入理解 Kotlin 協(xié)程》。(3)內(nèi)容系統(tǒng)深入:作者結(jié)合Kotlin編譯器源碼和工程實(shí)踐經(jīng)驗(yàn),從基礎(chǔ)知識、設(shè)計(jì)思想、技術(shù)方案、應(yīng)用方法、實(shí)踐技巧5個維度系統(tǒng)講解Kotlin元編程。(4)理論實(shí)戰(zhàn)兼?zhèn)洌翰粌H詳細(xì)講解了元編程的常見核心技術(shù),而且提供了大量來自真實(shí)生產(chǎn)環(huán)境的案例及代碼,圖文并茂。
Preface 前 言
為何寫作本書
2018年,我受邀在“JetBrains開發(fā)者日—2018中國巡演”活動中做題為“如何優(yōu)雅地使用Kotlin數(shù)據(jù)類”的分享。在準(zhǔn)備這次分享時,我花了幾天時間調(diào)研為Kotlin的數(shù)據(jù)類提供深復(fù)制能力的可行性,并給出了基于Kotlin反射和Java注解處理器(APT)的實(shí)現(xiàn)方案,也就是后來開源的DeepCopy項(xiàng)目。當(dāng)時我嘗試過編寫一款編譯器插件來實(shí)現(xiàn)這個需求,不過最終因?yàn)閷otlin編譯器的了解有限而未能如愿。
2021年,我受邀在Google開發(fā)者社區(qū)主辦的“社區(qū)說”活動中做題為“Kotlin編譯器插件:我們究竟在期待什么?”的分享。這一次,我花了兩周時間初步基于Kotlin符號處理器(KSP)和編譯器插件實(shí)現(xiàn)了數(shù)據(jù)類的深復(fù)制,整個過程充滿了探索的樂趣。
為了加深對Kotlin編譯器的認(rèn)識,我基于Kotlin編譯器插件完成了可以實(shí)現(xiàn)類似于Android的@IntDef功能的ValueDef編譯器插件。事實(shí)上,ValueDef的功能更強(qiáng)大,
@IntDef只會在代碼編寫時提供錯誤提示,而ValueDef除了會提供錯誤提示以外,還會在編譯時報錯。
與此同時,隨著Kotlin符號處理器的開源和Jetpack Compose的發(fā)布,大家對Kotlin元編程的關(guān)注度也在逐步提升,但這方面的相關(guān)資料非常少。
于是,我向機(jī)械工業(yè)出版社的楊福川老師提出了把這些內(nèi)容整理成書的想法,得到了他的肯定和支持。有了編寫《深入理解Kotlin協(xié)程》的經(jīng)驗(yàn),我很快就正式開始了這本書的寫作。
本書主要特點(diǎn)
“元編程”是一個比較龐大的話題,本書主要介紹了生產(chǎn)實(shí)踐中應(yīng)用較為廣泛的反射、Java注解處理器、Kotlin符號處理器、Kotlin編譯器插件、Kotlin語法分析等元編程相關(guān)的內(nèi)容。
與一般的語法知識不同,元編程相關(guān)的內(nèi)容通常較為抽象。為了更好地讓讀者理解元編程相關(guān)的各項(xiàng)技術(shù),本書提供了豐富的應(yīng)用案例。這些案例相對成熟和完善,可以作為元編程項(xiàng)目的范本。
本書基本上遵循了基礎(chǔ)知識介紹和案例實(shí)踐的結(jié)構(gòu)。以第3章為例,3.1節(jié)和3.2節(jié)系統(tǒng)地介紹了Java反射和Kotlin反射的概念和使用方法,是基礎(chǔ)知識介紹部分;3.3~3.5節(jié)通過案例進(jìn)一步介紹反射的適用場景,是案例實(shí)踐部分。
本書的實(shí)踐案例通常包括案例背景、需求分析和案例實(shí)現(xiàn)這幾方面。
案例背景:介紹案例的需求背景。本書的案例大多源自真實(shí)的生產(chǎn)實(shí)踐,因此案例背景的介紹有非常重要的價值。
需求分析:明確需求的細(xì)節(jié),拆解需求并轉(zhuǎn)換成技術(shù)方案。
案例實(shí)現(xiàn):提供詳細(xì)的問題解決思路以及案例實(shí)現(xiàn)步驟。
在系統(tǒng)介紹了常見的Kotlin元編程技術(shù)之后,本書還對Jetpack Compose的編譯器插件和IntelliJ插件、AtomicFU的字節(jié)碼JavaScript代碼邏輯做了詳細(xì)的剖析。
與絕大多數(shù)技術(shù)書類似,本書包含了大量代碼。為了提升閱讀體驗(yàn),我在編寫本書時對代碼做了以下優(yōu)化:
省略不必要的部分,避免代碼冗長而浪費(fèi)篇幅。
代碼縮進(jìn)為2個空格,以降低縮進(jìn)對閱讀體驗(yàn)的影響。
核心代碼注釋覆蓋率不低于30%,方便讀者快速理解代碼的含義。
核心代碼單行長度不超過80個字符,避免排版后出現(xiàn)折行的問題。
在部分代碼清單的開始處標(biāo)注其所在的模塊、文件或者函數(shù)等信息,方便讀者自行查找相關(guān)源代碼。
代碼字體采用JetBrains Mono,該字體由Kotlin項(xiàng)目團(tuán)隊(duì)所屬公司JetBrains為開發(fā)者專門打造,更適合代碼的閱讀。
本書閱讀對象
本書探討的內(nèi)容有一定的復(fù)雜度。在閱讀本書之前,讀者需要對Kotlin語言的語法有較為深入的理解,也需要具備一定的編譯原理的基礎(chǔ)知識。
本書適用于有一定基礎(chǔ)的Kotlin開發(fā)者,包括但不限于正在使用和希望使用Kotlin開發(fā)Android、Web服務(wù)、iOS、前端等應(yīng)用的開發(fā)者。
本書非常適用于希望在Kotlin相關(guān)開發(fā)領(lǐng)域?qū)崿F(xiàn)進(jìn)階的讀者。本書介紹的內(nèi)容對讀者提升自身編程水平以及團(tuán)隊(duì)提升研發(fā)效率都有非常大的參考價值。
本書不會介紹Kotlin的基礎(chǔ)語法,因此建議Kotlin初學(xué)者先閱讀相關(guān)基礎(chǔ)書。
如何閱讀本書
本書基于Kotlin 1.8.0系統(tǒng)地介紹了Kotlin元編程的基本概念、技術(shù)方案、應(yīng)用場景和實(shí)踐技巧。
本書主要分為三部分,分別介紹如下。
第一部分為元編程的基礎(chǔ)知識(第1章和第2章),為后續(xù)的元編程實(shí)踐提供知識儲備。如果讀者有一定的Kotlin元編程基礎(chǔ),可以直接閱讀第二部分內(nèi)容。在閱讀過程中,如果遇到概念相關(guān)的問題,也可以隨時翻閱這部分內(nèi)容。
第二部分為元編程的技術(shù)實(shí)踐(第3~8章),涉及運(yùn)行時的反射、源代碼生成、編譯時的符號處理、程序靜態(tài)分析、編譯器插件、元程序的開發(fā)和調(diào)試等內(nèi)容。這部分的章節(jié)安排相對獨(dú)立,讀者可以根據(jù)自己的實(shí)際需求選擇閱讀相應(yīng)的章節(jié)。需要說明的是,DeepCopy項(xiàng)目是貫穿這部分內(nèi)容的綜合案例,在介紹每一種元編程技術(shù)方案時,我們都會給出DeepCopy項(xiàng)目中對應(yīng)的技術(shù)方案的實(shí)現(xiàn),希望能夠幫助讀者加深對不同的元編程技術(shù)方案的認(rèn)識。在了解了元編程的常見技術(shù)之后,本書在第8章重點(diǎn)介紹了元編程項(xiàng)目實(shí)踐中編寫單元測試和集成測試的常見方法與技巧,以提升讀者開發(fā)元編程項(xiàng)目的
效率
目 錄 Contents
前言
第一部分 元編程的基礎(chǔ)知識
第1章 元編程概述2
1.1 元編程的需求背景2
1.2 元編程的基本概念4
1.2.1 元編程的定義5
1.2.2 元編程的分類5
1.3 元編程的學(xué)習(xí)方法6
1.3.1 培養(yǎng)興趣6
1.3.2 付諸行動6
1.3.3 善用工具7
1.3.4 多讀源代碼8
1.4 常用項(xiàng)目的調(diào)試環(huán)境配置8
1.4.1 Java編譯器8
1.4.2 Kotlin編譯器11
1.4.3 IntelliJ社區(qū)版13
1.4.4 Jetpack Compose編譯器插件19
1.5 本章小結(jié)21
第2章 元數(shù)據(jù)概述22
2.1 基本概念22
2.1.1 語法結(jié)構(gòu)23
2.1.2 編譯產(chǎn)物23
2.2 注釋23
2.2.1 注釋的結(jié)構(gòu)化23
2.2.2 文檔生成24
2.3 注解25
2.3.1 注解的概念25
2.3.2 源代碼可見的注解26
2.3.3 二進(jìn)制可見的注解27
2.3.4 運(yùn)行時可見的注解30
2.4 Kotlin的元數(shù)據(jù)31
2.4.1 Kotlin JVM中的@Metadata
注解31
2.4.2 Kotlin JVM模塊中的元數(shù)據(jù)35
2.4.3 klib中的元數(shù)據(jù)37
2.5 Kotlin的語法樹39
2.5.1 Kotlin的語法定義40
2.5.2 基于IntelliJ平臺接口的抽象語
法樹41
2.5.3 新一代語法樹FIR42
2.5.4 連接前后端編譯器的IR43
2.5.5 Java和Kotlin的符號樹45
2.6 Kotlin的編譯產(chǎn)物47
2.6.1 JVM47
2.6.2 JavaScript48
2.6.3 Native48
2.7 本章小結(jié)49
第二部分 元編程的技術(shù)實(shí)踐
第3章 運(yùn)行時的反射52
3.1 Java反射52
3.1.1 基本功能52
3.1.2 解除訪問限制53
3.1.3 動態(tài)代理54
3.1.4 對注解的支持55
3.1.5 對方法參數(shù)名的支持56
3.1.6 訪問Kotlin代碼57
3.2 Kotlin反射58
3.2.1 基本功能59
3.2.2 類引用的獲取61
3.2.3 屬性引用和函數(shù)引用65
3.2.4 typeOf67
3.2.5 dynamic類型69
3.2.6 屬性委托70
3.3 案例:Retrofit的接口實(shí)現(xiàn)72
3.3.1 Retrofit基本用法72
3.3.2 GitHubService實(shí)例的創(chuàng)建73
3.3.3 函數(shù)參數(shù)與請求參數(shù)的
對應(yīng)關(guān)系74
3.3.4 泛型類型的反序列化74
3.3.5 案例小結(jié)75
3.4 案例:使用反射實(shí)現(xiàn)DeepCopy75
3.4.1 案例背景75
3.4.2 需求分析76
3.4.3 案例實(shí)現(xiàn)78
3.4.4 小試牛刀79
3.4.5 案例小結(jié)79
3.5 案例:使用dynamic類型為
Kotlin JS實(shí)現(xiàn)DeepCopy80
3.5.1 案例背景80
3.5.2 需求分析80
3.5.3 案例實(shí)現(xiàn)83
3.5.4 案例小結(jié)83
3.6 本章小結(jié)84
第4章 源代碼生成85
4.1 直接輸出目標(biāo)代碼85
4.1.1 一個簡單的例子85
4.1.2 標(biāo)準(zhǔn)庫的代碼生成87
4.2 案例:為Kotlin添加Tuple類型88
4.2.1 案例背景88
4.2.2 需求分析90
4.2.3 案例實(shí)現(xiàn)91
4.3 使用模板引擎生成目標(biāo)代碼93
4.3.1 Anko中的代碼生成93
4.3.2 使用模板引擎渲染目標(biāo)代碼95
4.4 案例:為Java靜態(tài)方法生成
Kotlin擴(kuò)展函數(shù)(模板引擎)96
4.4.1 案例背景96
4.4.2 需求分析96
4.4.3 案例實(shí)現(xiàn)98
4.4.4 代碼優(yōu)化101
4.5 使用代碼生成框架生成目標(biāo)代碼104
4.5.1 JavaPoet104
4.5.2 KotlinPoet109
4.6 案例:為Java靜態(tài)方法生成
Kotlin擴(kuò)展函數(shù)(KotlinPoet)114
4.6.1 類型的映射114
4.6.2 實(shí)現(xiàn)代碼生成116
4.6.3 泛型參數(shù)的支持118
4.7 本章小結(jié)121
第5章 編譯時的符號處理122
5.1 符號的基本概念122
5.1.1 Java的符號122
5.1.2 Kotlin的符號124
5.1.3 符號與語法樹節(jié)點(diǎn)的關(guān)系和
區(qū)別125
5.2 處理器的基本結(jié)構(gòu)125
5.2.1 APT的基本結(jié)構(gòu)125
5.2.2 KSP的基本結(jié)構(gòu)130
5.2.3 APT與KSP的結(jié)構(gòu)差異131
5.2.4 處理器的配置文件132
5.3 深入理解符號和類型132
5.3.1 獲取修飾符133
5.3.2 通過名稱獲取符號133
5.3.3 獲取符號的類型134
5.3.4 通過類型獲取符號138
5.3.5 判斷類型之間的關(guān)系139
5.3.6 獲取注解及其參數(shù)值141
5.4 案例:基于源代碼生成模塊的
符號文件144
5.4.1 案例背景144
5.4.2 案例實(shí)現(xiàn):APT版本145
5.4.3 案例實(shí)現(xiàn):KSP版本147
5.5 深入理解符號處理器148
5.5.1 如何使用APT處理Kotlin
符號148
5.5.2 符號的有效性驗(yàn)證150
5.5.3 處理器的輪次和符號的延遲
處理150
5.5.4 處理器對增量編譯的支持151
5.5.5 多模塊的符號處理154
5.6 案例:使用符號處理器實(shí)現(xiàn)
DeepCopy156
5.6.1 案例背景156
5.6.2 需求分析156
5.6.3 案例實(shí)現(xiàn):APT版本157
5.6.4 案