溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Gradle自動(dòng)實(shí)現(xiàn)Android組件化詳解

發(fā)布時(shí)間:2020-07-16 14:11:03 來源:網(wǎng)絡(luò) 閱讀:399 作者:Android飛魚 欄目:移動(dòng)開發(fā)

為什么我們要用Gradle管理組件呢?

先來看看Android組件化需要實(shí)現(xiàn)的目標(biāo)

  1. 按照業(yè)務(wù)邏輯劃分模塊

  2. 項(xiàng)目模塊能夠單獨(dú)啟動(dòng)測試

  3. 能夠根據(jù)需求引入或刪除某些業(yè)務(wù)模塊

  4. 通過不同模塊的組合,組成不同的App

對于第一點(diǎn):需要根據(jù)技術(shù)架構(gòu)和業(yè)務(wù)架構(gòu)來劃分模塊,這里需要根據(jù)實(shí)際情況來考慮。我們需要優(yōu)化的是第二、三、四點(diǎn)。

對于第二點(diǎn):Android是通過應(yīng)用com.android.application或com.android.library來決定該模塊是以App模式還是以Library模式構(gòu)建。App模式和Library模式的最大區(qū)別就是,App能夠啟動(dòng),而Library不可以。所以如果我們的模塊能獨(dú)立啟動(dòng)的話,我們需要每次手動(dòng)去改動(dòng)模塊的build.gradle文件。好一點(diǎn)的做法定義一個(gè)布爾值來判斷是否處于debug模式,但是這里有個(gè)問題是,不是每個(gè)模塊都能獨(dú)立啟動(dòng)的。所以無論采用何種方案,都需要我們手動(dòng)管理。

對于第三點(diǎn):當(dāng)我們開發(fā)好業(yè)務(wù)模塊后,可能我們需要頻繁的新增或刪除某些業(yè)務(wù)模塊。如果是這樣的話,我們也是需要頻繁手動(dòng)修改App的build.gradle。

對于第四點(diǎn):有時(shí)候,我們可能會在不同的App中引用相同的組件(例如:滴滴的普通版和企業(yè)版,普通版包含企業(yè)版的功能),這個(gè)時(shí)候,我們也不希望要頻繁手動(dòng)管理組件依賴,特別是在組件還可以獨(dú)立運(yùn)行的時(shí)候。

所以,在我們實(shí)踐組件化的時(shí)候,最大的問題就是,我們需要頻繁的手動(dòng)build.gradle文件來管理組件應(yīng)用的插件和App的依賴。

使用Gradle來管理組件

先安利下筆者寫的Gradle插件:Calces。如果覺得這個(gè)插件有用的話,可以star下,如果你有更好的想法的話,可以向我提交pull request。

廢話少說,一下是通過Calces快速實(shí)現(xiàn)Android組件化構(gòu)建的流程。

Demo地址:SimpleCalces

項(xiàng)目結(jié)構(gòu):

Gradle自動(dòng)實(shí)現(xiàn)Android組件化詳解

  1. 引入依賴庫
    在Gradle 2.1及更高版本的插件構(gòu)建腳本代碼:
    在項(xiàng)目的build.gradle中

    buildscript?{
    ????...
    }
    plugins?{??id?"calces.modules"?version?"1.0.11"}

在較舊版本的Gradle中或需要?jiǎng)討B(tài)配置的情況下的插件構(gòu)建腳本代碼:

???buildscript?{
?????repositories?{
???????maven?{
?????????url?"https://plugins.gradle.org/m2/"
???????}
?????}
?????dependencies?{
???????classpath?"gradle.plugin.com.tangpj.tools:calces:1.0.11"
?????}
???}
???apply?plugin:?"calces.appConfig"

2.在項(xiàng)目build.gradle配置AppConfig

appConfig?{
????debugEnable?false

????apps?{
????????app{
????????????modules?':library1',?':library2'
????????}
????}

????modules{
????????library1{
????????????mainActivity?".Library1Activity"
????????????applicationId?"com.tangpj.library1"
????????????isRunAlone?true
????????}
????????library2{
????????????mainActivity?".Library2Activity"
????????????applicationId?"com.tangpj.library2"
????????????isRunAlone?true
????????}
????}

}

3.在modules(子模塊)引入模塊自動(dòng)化構(gòu)建插件 (注意:不需要手動(dòng)配置com.android.library或com.android.application)

apply?plugin:?'calces.modules'

這樣我們就完成了組件化的構(gòu)建了,是的,我們不再需要再手動(dòng)管理單個(gè)組件了與App的構(gòu)建了,通過Calces,我們能實(shí)現(xiàn)快速的組件化構(gòu)建與多App同時(shí)構(gòu)建。

那么問題來了,我們?nèi)绾螌?shí)現(xiàn)組件間的通信呢?在簡單的項(xiàng)目中,推薦該Demo一樣,通過使用Android隱式Intent來實(shí)現(xiàn)組件間通信。在中大型項(xiàng)目中,筆者推薦使用阿里的路由解決方案:ARouter。具體使用方法參考官方文檔就可以了,使用方法十分簡單。

注意:在使用隱式Intent實(shí)現(xiàn)組件件通信的時(shí)候需要注意找不到相應(yīng)組件異常:java.lang.IllegalStateException: Could not execute method of the activity。導(dǎo)致這個(gè)異常的原因是找不到目標(biāo)組件導(dǎo)致的,所以在實(shí)際開發(fā)的時(shí)候,需要捕獲這一異常,并且根據(jù)項(xiàng)目實(shí)際情況來進(jìn)行實(shí)際的處理。使用ARouter框架則能通過設(shè)置降級策略來實(shí)現(xiàn)異常處理(查看ARouter文檔了解更多)。

如果只是實(shí)現(xiàn)項(xiàng)目的簡單組件化,那么看到這里就可以了,如果希望實(shí)現(xiàn)更加靈活的組件化架構(gòu)的讀者可以繼續(xù)看下去,下面筆者將全面分析組件化的優(yōu)勢與筆者總結(jié)的組件化構(gòu)建思想。

組件化構(gòu)建簡述

組件化構(gòu)建與其說是一種技術(shù),不如說是一種思想。組件化構(gòu)建是通過對項(xiàng)目重新劃分成一個(gè)個(gè)高內(nèi)聚、低耦合的模塊來解決項(xiàng)目過于臃腫,代碼過于復(fù)雜的一種架構(gòu)思想。

我們通過對Google官方的架構(gòu)演示Demo todo-mvp進(jìn)行拆分來對Android組件化進(jìn)行深入的分析。

Demo地址:TodoCalces

todo系列app是Google?android-architecture項(xiàng)目中為了演示Android架構(gòu)的最佳實(shí)現(xiàn)而編寫的一系列演示Demo。todo app的特點(diǎn)是,它足夠簡單,代碼量少,易于理解。但是又不會過于簡單,因?yàn)樗且粋€(gè)包含完成功能的App。它實(shí)現(xiàn)了任務(wù)列表、任務(wù)詳情、新建任務(wù)、編輯任務(wù)、統(tǒng)計(jì)任務(wù)的功能。

todo-mvp實(shí)現(xiàn)的功能:

  • 任務(wù)列表

  • 任務(wù)詳情

  • 新增/編輯任務(wù)

  • 統(tǒng)計(jì)任務(wù)

我們將以todo-mvp的功能來劃分為4個(gè)業(yè)務(wù)模塊,將底層劃分為2個(gè)模塊,分別是superLib(提供mvp架構(gòu)支持與其它的一些支持庫功能)與dataLib(數(shù)據(jù)支持模塊,Room提供底層數(shù)據(jù)庫支持功能)。對于大型項(xiàng)目還可以加入resLib支持模塊,用來存放公共圖片資源、字符穿資源、樣式等。

架構(gòu)劃分圖如下:

Gradle自動(dòng)實(shí)現(xiàn)Android組件化詳解

從架構(gòu)圖可以看出,所有的業(yè)務(wù)組件都依賴底層庫,而APP又依賴于業(yè)務(wù)組件,APP組件在這里是作為一個(gè)獨(dú)立組件存在的。在一般的組件化實(shí)踐中,都不包含APP這個(gè)組件的,APP組件的存在是有其意義的。

首先,我們的組件化除了實(shí)現(xiàn)組件的獨(dú)立管理和動(dòng)態(tài)配置APP所依賴的組件外,還有一個(gè)十分重要的目的就是,通過組合不同的組件,打包多個(gè)不同的APP。例如,QQ有分普通版和輕聊版,輕聊版是功能簡化版的QQ。如果我們使用組件化來管理工程的話,我們只需要把不需要的模塊移除掉就可以了。而APP組件在這里的作用是充當(dāng)一個(gè)包裝盒,把需要的組件包裝進(jìn)來。并且我們可以通過控制包裝盒的樣式來配置不同的APP風(fēng)格。在這里我們可以通過Application中的Style來實(shí)現(xiàn)。

這里我們還是以todo-mvp為例,例如我們需要實(shí)現(xiàn)一個(gè)不包含統(tǒng)計(jì)功能的todo APP,按照我們的原理,我們只需要去掉statistics的依賴就可以了。

架構(gòu)劃分圖如下:

Gradle自動(dòng)實(shí)現(xiàn)Android組件化詳解

如果nostatsitcs需要不同的配色的方案的話,只需要在AndroidManifest的application標(biāo)簽中配置對應(yīng)的theme就可以了。

使用Calces實(shí)現(xiàn)todo-mvp的組件化

通過上面的分析,我們來試下對todo-mvp項(xiàng)目按照業(yè)務(wù)功能來劃分組件。我們先來看看劃分后的目錄:

Gradle自動(dòng)實(shí)現(xiàn)Android組件化詳解

好了,我們已經(jīng)對todo-mvp項(xiàng)目進(jìn)行初步的劃分了。根據(jù)上面分析的理論得知,我們的業(yè)務(wù)模塊是可以單獨(dú)運(yùn)行的,并且我們能夠快速構(gòu)建一個(gè)不包含statistics模塊的APP。

我們只需要使用Calces就能快速實(shí)現(xiàn)我們需要的功能。

按照Calces的教程,我們得知,實(shí)現(xiàn)Calces只需要三個(gè)步驟:

  1. 引入依賴庫

  2. 在項(xiàng)目的build.gradle中配置AppConfig

  3. 在業(yè)務(wù)模塊中引入模塊自動(dòng)化構(gòu)c持續(xù)

第一點(diǎn)和第三點(diǎn)在其它所有項(xiàng)目中的配置都是一樣的,在這里不作論述,下面我們看看對于TodoCalces項(xiàng)目,我們要如何配置AppConfig 。

appConfig?{

????debugEnable?false

????apps?{
????????app?{
????????????mainActivity?"com.tangpj.tasks.TasksActivity"
????????????modules?':modules:addtask',????????????????????':modules:taskdetail',????????????????????':modules:tasks',????????????????????':modules:statistics'
????????}

????????app2?{
????????????name?'nostatistic'
????????????applicationId?'com.tangpj.nostatistic'
????????????modules?':modules:addtask',????????????????????':modules:taskdetail',????????????????????':modules:tasks'
????????}

????}

????modules?{
????????addtask?{
????????????name?":modules:addtask"
????????????applicationId?"com.tangpj.addtask"
????????????mainActivity?".AddEditTaskActivity"
????????????isRunAlone?false
????????}

????????taskdetail?{
????????????name?":modules:taskdetail"
????????????applicationId?"com.tangpj.taskdetail"
????????????mainActivity?".TaskDetailActivity"
????????????isRunAlone?true
????????}

????????task?{
????????????name?":modules:tasks"
????????????applicationId?"com.tangpj.tasks"
????????????mainActivity?".TasksActivity"
????????????isRunAlone?true
????????}

????????statistics?{
????????????name?":modules:statistics"
????????????applicationId?"com.tangpj.statistics"
????????????mainActivity?".StatisticsActivity"
????????????isRunAlone?true
????????}

????}
}

根據(jù)AppConfig可以得出,我們分別配置了2個(gè)APP,分別是app1和app2。并且app2中是沒有依賴statistics的?,F(xiàn)在我們兩個(gè)APP運(yùn)行的對比圖。

app1(帶statistics模塊):

Gradle自動(dòng)實(shí)現(xiàn)Android組件化詳解

app2(不帶statistics模塊):

Gradle自動(dòng)實(shí)現(xiàn)Android組件化詳解

從運(yùn)行圖可以看出,app1和app2的配色方案是不一樣的,并且app2中不帶statistics模塊,通過對項(xiàng)目實(shí)行合理的劃分和引入Calces就能夠快速實(shí)現(xiàn)組件化構(gòu)建了。

結(jié)論:通過Calces能輕松實(shí)現(xiàn)業(yè)務(wù)組件的管理,多APP的快速構(gòu)建。當(dāng)我們的業(yè)務(wù)組件只有4個(gè)的時(shí)候,可能無法體現(xiàn)Calces的優(yōu)勢,但是如果我們的業(yè)務(wù)組件有40個(gè)的時(shí)候,Calces給我們帶來的優(yōu)勢就非常明顯了。我們可以通過靈活依賴不同的組件,實(shí)現(xiàn)快速構(gòu)建多個(gè)APP的目的。就像Calces的介紹圖案一樣,把組件當(dāng)成積木來使用。

如何測試

Android自動(dòng)化測試展開來說是一個(gè)非常大并且不算簡單的工程,在這里筆者不打算展開來說。只是簡單的介紹組件化構(gòu)建如何讓我們更方便地去測試。

并不是所有的自動(dòng)化測試都一樣,它們通常在使用范圍、實(shí)現(xiàn)難度和執(zhí)行時(shí)間上存在不同。我們一般把自動(dòng)化測試劃分為三種分別是:

  1. 單元測試:目的是測試代碼的最小單元。在基于Java的項(xiàng)目中,這個(gè)單元是一個(gè)方法。單元測試容易編寫,快速執(zhí)行,并在開發(fā)過程中針對代碼的正確性提供寶貴的反饋。

  2. 集成測試:用來測試一個(gè)完成的組件或子系統(tǒng),確保多個(gè)類之間的交互是否按預(yù)期運(yùn)行。集成測試需要比單元測試需要更長的執(zhí)行時(shí)間,而且更加難以維護(hù),失敗的原因難以診斷。

  3. 功能測試:通常用于測試應(yīng)用程序端到端的功能,包括從用戶的角度與所有外部系統(tǒng)的交互。當(dāng)我們討論用戶角度時(shí),通常是指用戶界面。因?yàn)橛脩艚缑鏁S著時(shí)間的推移發(fā)生變動(dòng),維護(hù)功能測試代碼會變得乏味而耗時(shí)。

為了優(yōu)化投資回報(bào)率,代碼庫應(yīng)該包含大量的單元測試、少量集成測試以及更少的功能測試。

占比如下圖所示:

Gradle自動(dòng)實(shí)現(xiàn)Android組件化詳解

從上文知道,在我們的組件化分的時(shí)候,會劃分一個(gè)基礎(chǔ)依賴庫(superLib)。基礎(chǔ)依賴庫為我們的項(xiàng)目提供了基本的支持,并且該庫在項(xiàng)目中是比較穩(wěn)定、并且不包含業(yè)務(wù)邏輯的,所以在基礎(chǔ)依賴庫中,我們應(yīng)該大量應(yīng)用單元測試。而集成測試則適用于我們的數(shù)據(jù)依賴庫(dataLib)中,我們可以通過集成測試來驗(yàn)證產(chǎn)品代碼與數(shù)據(jù)模塊的交互。而我們的業(yè)務(wù)模塊中包含了大量的業(yè)務(wù)邏輯,這部分是經(jīng)常變動(dòng)的部分,我們可以為我們的業(yè)務(wù)模塊編寫一些UI自動(dòng)化測試代碼,但是因?yàn)闃I(yè)務(wù)(界面)經(jīng)常變動(dòng)的原因,所以這部分測試代碼是難以維護(hù),并且復(fù)用性十分低的。。

最后,我們得出的結(jié)論是:應(yīng)該把主要精力放在單元測試上,所以如果當(dāng)你的精力不足以編寫所有測試代碼的時(shí)候,你應(yīng)該把主要的精力放在單元測試上,而不是放在收益最小的功能測試上。

關(guān)于自動(dòng)化測試,筆者給的建議就到這里了,如果需要深入理解測試的話,可以自行查找資料,或者關(guān)注筆者的博客。后續(xù)的博客中,有可能會寫關(guān)于自動(dòng)化測試相關(guān)的知識。

小結(jié)

通過Calces插件,我們在實(shí)現(xiàn)Android組件化時(shí)只需要關(guān)注如何合理劃分組件的架構(gòu)與如何實(shí)現(xiàn)組件間的通信就可以了。對于Android組件化來說,最主要問題有兩個(gè):

  1. 大型項(xiàng)目如何合理劃分組件模塊

  2. 當(dāng)項(xiàng)目的組件數(shù)量非常多的時(shí)候如何管理

第二個(gè)問題,可以通過Calces快速解決,至于第一個(gè)問題,筆者給出的指導(dǎo)就是,業(yè)務(wù)模塊在合理的情況下要盡可能的小,因?yàn)樵叫〉哪K,越容易達(dá)到高內(nèi)聚低耦合的目的。讀者不需要擔(dān)心項(xiàng)目模塊劃分得過于細(xì)不便于管理的問題,因?yàn)镃alces能夠輕松幫你管理好各個(gè)模塊。

感謝耐心看完。有不明白的對方歡迎留言交流

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI