AOP 全名 Aspect Oriented Programming扬蕊,意思是面向切面編程搀别。
AOP 跟 OOP 一樣,是一種編程思想尾抑。
如果 OOP 是縱向思想的話歇父,那么 AOP 就是橫向思想≡儆可以看下圖的差別:
OOP 與 AOP 的區(qū)別
面向目標不同:簡單來說 OOP 是面向名詞領(lǐng)域榜苫,AOP 面向動詞領(lǐng)域。
思想結(jié)構(gòu)不同:OOP 是縱向結(jié)構(gòu)翎冲,AOP 是橫向結(jié)構(gòu)垂睬。
注重方面不同:OOP 注重業(yè)務(wù)邏輯單元的劃分,AOP 偏重業(yè)務(wù)處理過程中的某個步驟或階段。
為什么要 AOP
在 Android 的架構(gòu)演進中驹饺,從 MVC 到 MVP 到 MVVM 再到 模塊化 和 組件化钳枕,可以看到每種架構(gòu)它們想解決的共同問題是如何更加精細地分離業(yè)務(wù)邏輯,并且處理好它們直接的關(guān)系赏壹,盡可能的將關(guān)注點集中在某一個模塊或者組件中鱼炒,所以可以這樣理解:
最佳的系統(tǒng)架構(gòu)由模塊化的關(guān)注面領(lǐng)域組成,每個關(guān)注面均用純 Java 對象(當然還可以用其他蝌借,這里強調(diào)的是用面向?qū)ο蟮乃枷耄崿F(xiàn)昔瞧。不同的領(lǐng)域之間用最不具有侵害性的「方面」或「類方面」工具整合起來(比如路由等)。
但是有一個問題需要我們反思:
雖然我們進行了模塊化或者組件化骨望,但是有很多模塊沒有做到恰當?shù)厍蟹株P(guān)注面硬爆,往往在業(yè)務(wù)邏輯中耦合了業(yè)務(wù)埋點、權(quán)限申請擎鸠、登陸狀態(tài)的判斷缀磕、對不可預(yù)知異常 try-catch 和一些持久化操作等等毫炉。
是否有一種方法可以把這些更好的解偶出來呢罐监,用 AOP 是一種不錯的選擇驶乾,「切分關(guān)注面」就是 AOP 的思想讯柔,它可以被看成是 OOP 的一種補充亩鬼。
AOP 的作用
AOP 的作用有很多本谜,舉例有 參數(shù)判空和校驗纹冤,權(quán)限驗證清酥,無埋點上報雄可,性能監(jiān)控凿傅,日志記錄,統(tǒng)一緩存代理数苫,
熱修改聪舒,異常捕獲,事件防抖虐急,事務(wù)處理等等箱残。
目前使用到 AOP 思想的比較有名的框架有:DataBinding,Dagger2止吁,ButterKnfie被辑,EventBus3,Room敬惦,DBFlow盼理,AndroidAnnotation,F(xiàn)ragmentRigger 等等俄删。
可見 AOP 的作用是挺大的榜揖。
AOP 的實現(xiàn)方式
動態(tài)織入 Hook 方式
在運行期勾哩,目標類加載后,為接口動態(tài)生成代理類举哟,將切面植入到代理類中思劳。相對于靜態(tài)AOP更加靈活。但切入的關(guān)注點需要實現(xiàn)接口妨猩。
實現(xiàn)方式有:Dexposed潜叛,Xposed,epic
動態(tài)字節(jié)碼生成
原理是在運行期間目標字節(jié)碼加載后壶硅,通過字節(jié)碼技術(shù)為一個類創(chuàng)建子類威兜,并在子類中采用方法攔截的技術(shù)攔截所有父類方法的調(diào)用,順勢織入橫切邏輯庐椒。由于是通過子類來代理父類椒舵,因此不能代理被 final 字段修飾的方法。
實現(xiàn)方式有:Cglib + Dexmaker
靜態(tài)織入方式
- 在編譯期織入约谈,切面直接以字節(jié)碼的形式編譯到目標字節(jié)碼文件中笔宿,這要求使用特殊的 Java 編譯器。
- 在類裝載期織入棱诱,這要求使用特殊的類裝載器泼橘。
實現(xiàn)方式有:APT,AspectJ迈勋,ASM炬灭,Javassist,DexMaker靡菇,ASMDEX
通過一張圖理解他們的關(guān)系:
他們的區(qū)別就是插入的時機不同重归。
常用的 AOP 實現(xiàn)方法有 APT,AspectJ厦凤,Javassist鼻吮。其他相對來說比較不常用。
APT
說到 APT 泳唠,那就離不開注解狈网,Android 解析注解主要有兩種實現(xiàn)宙搬。
第一種是運行期通過反射去解析當前類笨腥,注入相應(yīng)要運行的方法。這種方法最大的缺點就是性能問題勇垛,因為用了反射脖母。大量使用會導(dǎo)致 app 性能下降,比如早期的 butterknife 框架闲孤。
第二種是通過編譯器生成類的代理類谆级,在運行期直接調(diào)用代理類的代理方法烤礁。這種方法的優(yōu)點就是沒有性能問題,缺點可能就是如果使用很多會生成比較多的代理類肥照,帶上控制得當?shù)脑拺?yīng)該沒什么問題脚仔,比如現(xiàn)在的 butterknife,EventBus3舆绎,Room 框架等鲤脏。
APT 就是指第二種方法。
在編寫 APT 代理類時吕朵,往往有自動生成源代碼的需要猎醇,我們不可能自己手動拼接字符串,這時候可以用 J 神的 JavaPoet 框架去解決努溃。
AspectJ
AspectJ 是目前來說比較好的 AOP 框架了硫嘶,它完全兼容 Java,而且使用起來比較簡單梧税。
但在 Android 上集成比較麻煩沦疾,這時候可以通過一些大神們提供的插件解決:
- J 神的 Hugo
- gradle_plugin_android_aspectjx
Javassist
使用 Javassist 的代表框架有 熱修復(fù)框架HotFix 、Savior(InstantRun)
Javassist 是一個編輯字節(jié)碼的框架贡蓖,作用是修改編譯后的 class 字節(jié)碼
實現(xiàn) Javassist 的方式可以通過自定義 gradle 插件曹鸠,利用 Gradle Transfrom 這個 api 去實現(xiàn)。原因是 Transform 更為方便斥铺,我們不再需要插入到某個Task前面彻桃。Tranfrom 有自己的執(zhí)行時機,一經(jīng)注冊便會自動添加到 Task 執(zhí)行序列中晾蜘,且正好是 class 被打包成 dex 之前邻眷。