文章目录

  • 依赖的方式
  • 直接依赖
  • 项目依赖
  • 本地jar包依赖
  • 依赖的类型
  • api和implementation的区别
  • 依赖冲突及解决方案
  • 移除某个依赖
  • 不允许依赖传递
  • 强制使用某个版本
  • 依赖冲突时立刻构建失败


依赖的方式

Gradle中的依赖方式有直接依赖、项目依赖和本地jar包依赖三种:

dependencies {
    // 1、依赖当前项目下的某个模块[子工程]
    implementation project(':subject01')
    // 2、直接依赖本地的某个jar文件
    implementation files('libs/foo.jar', 'libs/bar.jar')
    // 2、配置某文件夹作为依赖项
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    // 3、直接依赖
    implementation 'org.apache.logging.log4j:log4j:2.17.2'
}

直接依赖

在项目中直接导入的依赖,即为直接依赖,如:

implementation 'org.apache.logging.log4j:log4j:2.17.2'

完整写法如下,其中group/name/version共同定位一个远程jar包:

implementation group: 'org.apache.logging.log4j', name: 'log4j;, version: '2.17.2'

项目依赖

依赖项目中的另一个模块,被依赖的模块需要是library模块,并且在settings.gradle中配置:

implementation project(':subject01')

本地jar包依赖

即依赖本地jar包,有如下两种方式:

// 2、直接依赖本地的某个jar文件
    implementation files('libs/foo.jar', 'libs/bar.jar')
    // 2、配置某文件夹作为依赖项
    implementation fileTree(dir: 'libs', include: ['*.jar'])

依赖的类型

gradle项目添加mysql gradle项目添加依赖_gradle


其中java插件的功能,java-library插件都提供。

api和implementation的区别

如下表所示:

api

implementation

编译时

能进行依赖传递;底层变,上层全部都要变;编译速度慢

不能进行依赖传递;底层变,上层不会变化;编译速度快

运行时

运行时会加载,所有模块的类都会被加载

运行时会加载,所有模块的类都会被加载

应用场景

适用于多模块依赖,避免重复依赖

多数情况下使用implementation

以下图为例:

gradle项目添加mysql gradle项目添加依赖_gradle项目添加mysql_02


libC发生变化时,libAprojectX也随之变化,都需要重新编译;当libD发生变化时,直接依赖它的libB随之变化,而没有直接依赖libDprojectX不会发生变化,也只有libDlibB要重新编译。

再考虑一种场景:一个工程中,moduleA依赖moduleBmoduleCmoduleB也依赖moduleC,因此可以让moduleBapi的方式依赖moduleCmoduleA则只implementation依赖moduleB即可。

再例如,一个工程中有ABCD四个模块:
1)、A implmentation B,B implementation C,则A不可用C;
2)、A implmentation B,B api C,则A可用C;
3)、A implmentation B,B implementation C,C api D,则B可用D,A不可用D;
4)、A implmentation B,B api C,C api D,则A可用D。

任何情况下,发生依赖的模块里所有的类都会被加载。

依赖冲突及解决方案

依赖冲突是指,在编译过程中,若存在对某个包的多版本依赖,构建系统要选择哪个进行构建,如下图所示:

gradle项目添加mysql gradle项目添加依赖_gradle_03


其中,ABC都是本地项目或模块,log4j时远程依赖。编译时,BC各用各的log4j,彼此没有冲突。但打包时,只能有一个版本的代码被打到jar包中,因此就发生了冲突。

事实上,gradle默认会选择最新的版本去打包,因为新版本的jar包一般都是向下兼容的,因此推荐这种官方的默认解决方法,不过gradle也提供了一系列的解决依赖冲突的方法:移除某个依赖;不允许依赖传递或强制使用某个版本。

移除某个依赖

dependencies {
    testImplementation 'org.junit.jupiter:junit-jupiter-api:5.8.1'
    testRuntimeOnly 'org.junit.jupiter:junit-jupiter-engine:5.8.1'

    implementation('org.hibernate:hibernate-core:3.6.3.Final'){
        // 排除某一个库(slf4j)依赖:如下三种写法都行
        exclude group: 'org.slf4j'
        exclude module: 'slf4j-api'
        exclude group: 'org.slf4j',module: 'slf4j-api'
    
    // 排除之后,使用手动的引入即可。
    implementation 'org.slf4j:slf4j-api:1.4.0'
}

不允许依赖传递

dependencies {
    testImplementation 'org.junit.jupiter:junit-jupiter-api:5.8.1'
    testRuntimeOnly 'org.junit.jupiter:junit-jupiter-engine:5.8.1'
    implementation('org.hibernate:hibernate-core:3.6.3.Final'){
        // 不允许依赖传递,一般不用
        transitive(false)
    }
    // 排除之后,使用手动的引入即可
    implementation 'org.slf4j:slf4j-api:1.4.0'
}

不允许依赖传递,则该依赖的所有内部依赖均不会添加到编译或运行时的类路径中,

强制使用某个版本

dependencies {
    testImplementation 'org.junit.jupiter:junit-jupiter-api:5.8.1'
    testRuntimeOnly 'org.junit.jupiter:junit-jupiter-engine:5.8.1'
    implementation('org.hibernate:hibernate-core:3.6.3.Final')
    // 强制使用某个版本【官方建议使用这种方式】
    implementation('org.slf4j:slf4j-api:1.4.0!!')
    // 这种效果和上面那种一样,强制指定某个版本
    implementation('org.slf4j:slf4j-api:1.4.0'){
        version{
            strictly("1.4.0")
        }
    }
}

依赖冲突时立刻构建失败

事实上,我们可以配置当Gradle遇到依赖冲突时,立刻构建失败,从而找出项目或模块中的所有的依赖冲突:

// 项目或模块的build.gradle

configurations.all {
    Configuration config -> {
        // 当遇到版本冲突时直接构建失败
        config.resolutionStrategy.failOnVersionConflict()
    }
}