之前对Android Gradle构建的依赖一直傻傻分不清,这段时间正好接入集团的一个二方库,踩了很多坑,也顺带把Gradle依赖这块搞清楚了,主要整理了下Gradle依赖的类型、依赖配置、如何查看依赖、依赖冲突如何解决。
dependencies DSL标签是标准Gradle API中的一部分,而不是Android Gradle插件的特性,所以它不属于android标签。
依赖有三种方式,如下面的例子:
- apply plugin: 'com.android.application'
-
- android { ... }
-
- dependencies {
- // Dependency on a local library module
- implementation project(":mylibrary")
-
- // Dependency on local binaries
- implementation fileTree(dir: 'libs', include: ['*.jar'])
-
- // Dependency on a remote binary
- implementation 'com.example.android:app-magic:12.3'
- }
-
- implementation project(":mylibrary")
-
这种依赖方式是直接依赖本地库工程代码的(需要注意的是,mylibrary的名字必须匹配在settings.gradle中include标签下定义的模块名字)。
- implementation fileTree(dir: 'libs', include: ['*.jar'])
-
这种依赖方式是依赖工程中的 module_name/libs/目录下的Jar文件(注意Gradle的路径是相对于build.gradle文件来读取的,所以上面是这样的相对路径)。
如果只想依赖单个特定本地二进制库,可以如下配置:
- implementation files('libs/foo.jar', 'libs/bar.jar')
-
- implementation 'com.example.android:app-magic:12.3'
-
上面是简写的方式,这种依赖完整的写法如下:
- implementation group: 'com.example.android', name: 'app-magic', version: '12.3'
-
group、name、version共同定位一个远程依赖库。需要注意的点是,version最好不要写成"12.3+"这种方式,除非有明确的预期,因为非预期的版本更新会带来构建问题。远程依赖需要在repositories标签下声明远程仓库,例如jcenter()、google()、maven仓库等。
目前Gradle版本支持的依赖配置有:implementation、api、compileOnly、runtimeOnly和annotationProcessor,已经废弃的配置有:compile、provided、apk、providedCompile。此外依赖配置还可以加一些配置项,例如AndroidTestImplementation、debugApi等等。
常用的是implementation、api、compileOnly三个依赖配置,含义如下:
可以查看单个module或者这个project的依赖,通过运行依赖的Gradle任务,如下:
随着很多依赖加入到项目中,难免会出现依赖冲突,出现依赖冲突如何解决?
依赖冲突可能会报类似下面的错误:
- Program type already present com.example.MyClass
-
通过查找类的方式(command + O)定位到冲突的依赖,进行排除。
- compile('com.taobao.android:accs-huawei:1.1.2@aar') {
- transitive = true
- exclude group: 'com.taobao.android', module: 'accs_sdk_taobao'
- }
-
- configurations {
- compile.exclude module: 'cglib'
- //全局排除原有的tnet jar包与so包分离的配置,统一使用aar包中的内容
- all*.exclude group: 'com.taobao.android', module: 'tnet-jni'
- all*.exclude group: 'com.taobao.android', module: 'tnet-so'
- }
-
- compile('com.zhyea:ar4j:1.0') {
- transitive = false
- }
-
- configurations.all {
- transitive = false
- }
-
还可以在单个依赖项中使用@jar标识符忽略传递依赖:
- compile 'com.zhyea:ar4j:1.0@jar'
-
- compile('com.zhyea:ar4j:1.0') {
- force = true
- }
-
- configurations.all {
- resolutionStrategy {
- force 'org.hamcrest:hamcrest-core:1.3'
- }
- }
-
- task zip(type: Zip) {
- into('lib') {
- from(configurations.runtime) {
- exclude '*unwanted*', '*log*'
- }
- }
- into('') {
- from jar
- from 'doc'
- }
- }
-
代码表示在打zip包的时候会过滤掉名称中包含“unwanted”和“log”的jar包。这里调用的exclude方法的参数和前面的例子不太一样,前面的参数多是map结构,这里则是一个正则表达式字符串。
也可以使用在打包时调用include方法选择只打包某些需要的依赖项:
- task zip(type: Zip) {
- into('lib') {
- from(configurations.runtime) {
- include '*ar4j*', '*spring*'
- }
- }
- into('') {
- from jar
- from 'doc'
- }
- }
-
主要是使用dependencies中排除和全局配置排除。