在app/build.gradle
有以下库依赖:api
dependencies {
compile "com.xxx.yyy:somelib:$version"
}
复制代码
如今若是要改为源码依赖(工程依赖)须要改2处地方: app/build.gradle
:bash
dependencies {
compile project(':somelib')
}
复制代码
settings.gradle
:app
include ':somelib'
project(":somelib").projectDir = new File("/lib/dir", "/some/path/somelib")
复制代码
如今依赖的库很是多, 并且还存在依赖嵌套, 但愿只改一处配置就可以切换库依赖与源码依赖gradle
显然用库依赖settings.gradle
里不须要include
那坨语句, 那么必然须要在settings.gradle
里定义一个变量来做条件判断, 这里须要解决的一个核心问题是app/build.gradle
可以访问到settings.gradle
里的这个变量来继续来做compile "com.xxx.yyy:somelib:$version"
仍是compile project(':somelib')
的判断, 也就是须要一个***全局可访问的gradle变量***ui
网上搜索了一圈居然没找到!this
研究了一下发现如下重要2点:spa
settings.gradle
对象生成早于app/build.gradle
甚至早于根目录的build.gradle
, 因此在build.gradle里声明ext { someVar=xxx }
变量无效, settings没法访问app/build.gradle
上下文依赖的project
对象与settings.gradle
共享同一个gradle对象因而有以下配置: settings.gradle
:code
def module_config = [
'somelib': '',
'aaa': '1.0.0',
'bbb': '0.2.0',
'ccc': '0.0.3',
]
module_config.each { k, v ->
if (!v) {
include ":${k}"
project(":${k}").projectDir = new File(rootDir.getParent(), "/some/path/${k}")
}
}
gradle.ext.dependon = { project, name ->
def ver = module_config[name]
def handler = project.dependencies
if (!ver) {
handler.compile project.project(":$name")
} else {
handler.compile "com.xxx.yyy:$name:$ver"
}
}
复制代码
这里用gradle.ext
定义了一个全局的方法而不是一个变量, 这样在引用的时候直接调用: app/build.gradle
:对象
dependencies {
gradle.ext.dependon(this, "somelib")
gradle.ext.dependon(this, "aaa")
gradle.ext.dependon(this, "ccc")
}
复制代码
这样咱们只需更改module_config
中对应模块的版本就能够实现一动态改变工程依赖的功能了!ci
注意gradle.ext.dependon
传入的是org.gradle.apiProject
对象, 而project.dependencies是org.gradle.api.artifacts.dsl.DependencyHandler
对象, 分别是gradle配置类
最根本最重要最核心的实际上是gradle.ext
这个全局可访问变量, 只要有了它彻底能够用任何方式实现.