Gradle 依賴切換源碼實踐示例詳解
引言
最近,因為開發(fā)的時候經(jīng)改動依賴的庫,所以,我想對 Gradle 腳本做一個調(diào)整,用來動態(tài)地將依賴替換為源碼。這里以 android-mvvm-and-architecture 這個工程為例。該工程以依賴的形式引用了我的另一個工程 AndroidUtils。在之前,當我需要對 AndroidUtils 這個工程源碼進行調(diào)整時,一般來說有兩種解決辦法。
1、一般的修改辦法
一種方式是,直接修改 AndroidUtils 這個項目的源碼,然后將其發(fā)布到 MavenCentral. 等它在 MavenCentral 中生效之后,再將項目中的依賴替換為最新的依賴。這種方式可行,但是修改的周期太長。
另外一種方式是,修改 Gradle 腳本,手動地將依賴替換為源碼依賴。此時,需要做幾處修改,
修改 1,在 settings.gradle 里面將源碼作為子工程添加到項目中,
include ':utils-core', ':utils-ktx' project(':utils-core').projectDir = new File('../AndroidUtils/utils') project(':utils-ktx').projectDir = new File('../AndroidUtils/utils-ktx')
修改 2,將依賴替換為工程引用,
// implementation "com.github.Shouheng88:utils-core:$androidUtilsVersion" // implementation "com.github.Shouheng88:utils-ktx:$androidUtilsVersion" // 上面的依賴替換為下面的工程引用 implementation project(":utils-core") implementation project(":utils-ktx")
這種方式亦可行,只不過過于繁瑣,需要手動修改 Gradle 的構(gòu)建腳本。
2、通過 Gradle 腳本動態(tài)修改依賴
其實 Gradle 是支持動態(tài)修改項目中的依賴的。動態(tài)修改依賴在上述場景,特別是組件化的場景中非常有效。這里我參考了公司組件化的切換源碼的實現(xiàn)方式,用了 90 行左右的代碼就實現(xiàn)了上述需求。
2.1 配置文件和工作流程抽象
這種實現(xiàn)方式里比較重要的一環(huán)是對切換源碼工作機制的抽象。這里我重新定義了一個 json 配置文件,
[ { "name": "AndroidUtils", "url": "git@github.com:Shouheng88/AndroidUtils.git", "branch": "feature-2.8.0", "group": "com.github.Shouheng88", "open": true, "children": [ { "name": "utils-core", "path": "AndroidUtils/utils" }, { "name": "utils-ktx", "path": "AndroidUtils/utils-ktx" } ] } ]
它內(nèi)部的參數(shù)的含義分別是,
name
:工程的名稱,對應于 Github 的項目名,用于尋找克隆到本地的代碼源碼url
:遠程倉庫的地址branch
:要啟用的遠程倉庫的分支,這里我強制自動切換分支時的本地分支和遠程分支同名group
:依賴的 group idopen
:表示是否啟用源碼依賴children.name
:表示子工程的 module 名稱,對應于依賴中的artifact id
children.path
:表示子工程對應的相對目錄
也就是說,
- 一個工程下的多個子工程的
group id
必須相同 children.name
必須和依賴的artifact id
相同
上述配置文件的工作流程是,
def sourceSwitches = new HashMap<String, SourceSwitch>() // Load sources configurations. parseSourcesConfiguration(sourceSwitches) // Checkout remote sources. checkoutRemoteSources(sourceSwitches) // Replace dependencies with sources. replaceDependenciesWithSources(sourceSwitches)
- 首先,Gradle 在 setting 階段解析上述配置文件
- 然后,根據(jù)解析的結(jié)果,將打開源碼的工程通過 project 的形式引用到項目中
- 最后,根據(jù)上述配置文件,將項目中的依賴替換為工程引用
2.2 為項目動態(tài)添加子工程
如上所述,這里我們忽略掉 json 配置文件解析的環(huán)節(jié),直接看拉取最新分支并將其作為子項目添加到項目中的邏輯。該部分代碼實現(xiàn)如下,
/** Checkout remote sources if necessary. */ def checkoutRemoteSources(sourceSwitches) { def settings = getSettings() def rootAbsolutePath = settings.rootDir.absolutePath def sourcesRootPath = new File(rootAbsolutePath).parent def sourcesDirectory = new File(sourcesRootPath, "open_sources") if (!sourcesDirectory.exists()) sourcesDirectory.mkdirs() sourceSwitches.forEach { name, sourceSwitch -> if (sourceSwitch.open) { def sourceDirectory = new File(sourcesDirectory, name) if (!sourceDirectory.exists()) { logd("clone start [$name] branch [${sourceSwitch.branch}]") "git clone -b ${sourceSwitch.branch} ${sourceSwitch.url} ".execute(null, sourcesDirectory).waitFor() logd("clone completed [$name] branch [${sourceSwitch.branch}]") } else { def sb = new StringBuffer() "git rev-parse --abbrev-ref HEAD ".execute(null, sourceDirectory).waitForProcessOutput(sb, System.err) def currentBranch = sb.toString().trim() if (currentBranch != sourceSwitch.branch) { logd("checkout start current branch [${currentBranch}], checkout branch [${sourceSwitch.branch}]") def out = new StringBuffer() "git pull".execute(null, sourceDirectory).waitFor() "git checkout -b ${sourceSwitch.branch} origin/${sourceSwitch.branch}" .execute(null, sourceDirectory).waitForProcessOutput(out, System.err) logd("checkout completed: ${out.toString().trim()}") } } // After checkout sources, include them as subprojects. sourceSwitch.children.each { child -> settings.include(":${child.name}") settings.project(":${child.name}").projectDir = new File(sourcesDirectory, child.path) } } } }
這里,我將子項目的源碼克隆到 settings.gradle
文件的父目錄下的 open_sources
目錄下面。這里當該目錄不存在的時候,我會先創(chuàng)建該目錄。這里需要注意的是,我在組織項目目錄的時候比較喜歡將項目的子工程放到和主工程一樣的位置。所以,上述克隆方式可以保證克隆到的 open_sources
仍然在當前項目的工作目錄下。
然后,我對 sourceSwitches
,也就是解析的 json 文件數(shù)據(jù),進行遍歷。這里會先判斷指定的源碼是否已經(jīng)拉下來,如果存在的話就執(zhí)行 checkout 操作,否則執(zhí)行 clone 操作。這里在判斷當前分支是否為目標分支的時候使用了 git rev-parse --abbrev-ref HEAD
這個 Git 指令。該指令用來獲取當前倉庫所處的分支。
最后,將源碼拉下來之后通過 Settings
的 include()
方法加載指定的子工程,并使用 Settings
的 project()
方法指定該子工程的目錄。這和我們在 settings.gradle
文件中添加子工程的方式是相同的,
include ':utils-core', ':utils-ktx' project(':utils-core').projectDir = new File('../AndroidUtils/utils') project(':utils-ktx').projectDir = new File('../AndroidUtils/utils-ktx')
2.3 使用子工程替換依賴
動態(tài)替換工程依賴使用的是 Gradle 的 ResolutionStrategy 這個功能。也許你對諸如
configurations.all { resolutionStrategy.force 'io.reactivex.rxjava2:rxjava:2.1.6' }
這種寫法并不陌生。這里的 force
和 dependencySubstitution
一樣,都屬于 ResolutionStrategy 提供的功能的一部分。只不過這里的區(qū)別是,我們需要對所有的子項目進行動態(tài)更改,因此需要等項目 loaded 完成之后才能執(zhí)行。
下面是依賴替換的實現(xiàn)邏輯,
/** Replace dependencies with sources. */ def replaceDependenciesWithSources(sourceSwitches) { def gradle = settings.gradle gradle.projectsLoaded { gradle.rootProject.subprojects { configurations.all { resolutionStrategy.dependencySubstitution { sourceSwitches.forEach { name, sourceSwitch -> sourceSwitch.children.each { child -> substitute module("${sourceSwitch.artifact}:${child.name}") with project(":${child.name}") } } } } } } }
這里使用 Gradle 的 projectsLoaded
這個點進行 hook,將依賴替換為子工程。
此外,也可以將子工程替換為依賴,比如,
dependencySubstitution { substitute module('org.gradle:api') using project(':api') substitute project(':util') using module('org.gradle:util:3.0') }
2.4 注意事項
上述實現(xiàn)方式要求多個子工程的腳本盡可能一致。比如,在 AndroidUtils 的獨立工程中,我通過 kotlin_version
這個變量指定 kotlin 的版本,但是在 android-mvvm-and-architecture 這個工程中使用的是 kotlinVersion
. 所以,當切換了子工程的源碼之后就會發(fā)現(xiàn) kotlin_version
這個變量找不到了。因此,為了實現(xiàn)可以動態(tài)切換源碼,是需要對 Gradle 腳本做一些調(diào)整的。
在我的實現(xiàn)方式中,我并沒有將子工程的源碼放到主工程的根目錄下面,也就是將 open_sources
這個目錄放到 appshell 這個目錄下面。而是放到和 appshell 同一級別。
這樣做的原因是,實際開發(fā)過程中,通常我們會克隆很多倉庫到 open_sources
這個目錄下面(或者之前開發(fā)遺留下來的克隆倉庫)。有些倉庫雖然我們關閉了源碼依賴,但是因為在 appshell 目錄下面,依然會出現(xiàn)在 Android Studio 的工程目錄里。而按照上述方式組織目錄,我切換了哪個項目等源碼,哪個項目的目錄會被 Android Studio 加載。其他的因為不在 appshell 目錄下面,所以會被 Android Studio 忽略。這種組織方式可以盡可能減少 Android Studio 加載的文本,提升 Android Studio 響應的速率。
總結(jié)
上述是開發(fā)過程中替換依賴為源碼的“無痕”修改方式。不論在組件化還是非組件化需要開發(fā)中都是一種非常實用的開發(fā)技巧。按照上述開發(fā)開發(fā)方式,我們可以既能開發(fā) android-mvvm-and-architecture 的時候隨時隨地打開 AndroidUtils 進行修改,亦可對 AndroidUtil 這個工程獨立編譯和開發(fā)。
源代碼參考 android-mvvm-and-architecture 項目(當前是 feature-3.0 分支)的 AppShell 下面的 sources.gradle
文件。
以上就是Gradle 依賴切換源碼實踐示例詳解的詳細內(nèi)容,更多關于Gradle 依賴切換的資料請關注腳本之家其它相關文章!
相關文章
Android Drawerlayout側(cè)拉欄事件傳遞問題的解決方法
這篇文章主要為大家詳細介紹了Android Drawerlayout側(cè)拉欄事件傳遞問題的解決方法,具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-11-11Android中activity從創(chuàng)建到顯示的基本介紹
這篇文章主要給大家介紹了關于Android中activity從創(chuàng)建到顯示的相關資料,文中通過示例代碼介紹的非常詳細,對各位Android初學者們具有一定的參考學習價值,需要的朋友們下面隨著小編來一起看看吧。2017-11-11Android RecyclerView實現(xiàn)拼團倒計時列表實例代碼
這篇文章主要給大家介紹了關于Android RecyclerView實現(xiàn)拼團倒計時列表的相關資料,文中通過示例代碼介紹的非常詳細,對各位Android開發(fā)者們具有一定的參考學習價值,需要的朋友們下面來一起學習學習吧2019-09-09Android實現(xiàn)在列表List中顯示半透明小窗體效果的控件用法詳解
這篇文章主要介紹了Android實現(xiàn)在列表List中顯示半透明小窗體效果的控件用法,結(jié)合實例形式分析了Android半透明提示框的實現(xiàn)與設置技巧,需要的朋友可以參考下2016-06-06Android和JavaScript相互調(diào)用的方法
這篇文章主要介紹了Android和JavaScript相互調(diào)用的方法,實例分析了Android的WebView執(zhí)行JavaScript及JavaScript訪問Android的技巧,需要的朋友可以參考下2015-12-12Android自定義SwipeLayout仿QQ側(cè)滑條目
這篇文章主要為大家詳細介紹了Android自定義SwipeLayout仿QQ側(cè)滑條目,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-08-08Android如何跳轉(zhuǎn)到應用商店的APP詳情頁面
最近做項目遇到這樣的需求,要求從App內(nèi)部點擊按鈕或鏈接,跳轉(zhuǎn)到應用商店的某個APP的詳情頁面,怎么實現(xiàn)此功能呢?下面小編給大家分享Android如何跳轉(zhuǎn)到應用商店的APP詳情頁面,需要的朋友參考下2017-01-01