项目依赖统一管理方案

一般对项目的依赖管理会有以下需求。

1、项目依赖同意管理,在单独文件中配置。

2、不同的Module种的依赖版本号统一

3、不同项目种的依赖版本号统一

针对这些需求,提出了以下集中方案:

1、使用循环优化Gradle依赖管理

2、使用buildSrc管理Gradle依赖

3、使用includeBuild统一配置依赖版本

4、使用Version Catalog统一配置依赖版本。

使用循环优化Gradle依赖管理

这种方式就是在app的目录下创建一个xxx.gradle文件,然后将我们项目所有需要用到的依赖放到该文件里,并分好类,例如:

ext{
    android = [
            compileSdkVersion       : 29,
            buildToolsVersion       : "29.0.3",
            minSdkVersion           : 21,
            targetSdkVersion        : 29,
            versionCode             : 10101,
            versionName             : "1.01.01"
    ]
    dependencies  = [
            "rxlifecycle"                 : "com.trello.rxlifecycle2:rxlifecycle:2.2.2",
            "rxlifecycle-android"         : "com.trello.rxlifecycle2:rxlifecycle-android:2.2.2",
            "rxlifecycle-components"      : "com.trello.rxlifecycle2:rxlifecycle-components:2.2.2",
            //网络
            "retrofit"                    : "com.squareup.retrofit2:retrofit:2.6.2",
            "retrofit_adapter"            : "com.squareup.retrofit2:adapter-rxjava2:2.6.2",
            "retrofit_gson"               : "com.squareup.retrofit2:converter-gson:2.6.2",
            "rxjava"                      : "io.reactivex.rxjava2:rxjava:2.2.14",
            "rx_android"                  : "io.reactivex.rxjava2:rxandroid:2.1.1",
            //gson
          
            ........

    ]

    annotationProcessor = [
            "butterknife-compiler"         :"com.jakewharton:butterknife-compiler:10.0.0",
            "glide-compiler"              : "com.github.bumptech.glide:compiler:4.9.0",
            "dagger-compiler"             : "com.google.dagger:dagger-compiler:2.24",
    ]
    
    apiFileDependencies = [
            "launchstarter"                                   :"libs/launchstarter-release-1.0.0.aar"
    ]
    
    debugImplementationDependencies = [
            "MethodTraceMan"                                  : "com.github.zhengcx:MethodTraceMan:1.0.7"
    ]
    
    ...
    
    implementationExcludes = [
            "com.android.support.test.espresso:espresso-idling-resource:3.0.2" : [
                    'com.android.support' : 'support-annotations'
            ]
    ]

然后在我们的build.gradle文件中引用该文件

apply from:"config.gradle"

最后在build.gradle的dependencies中添加如下代码:

def implementationDependencies = project.ext.dependencies
def processors = project.ext.annotationProcesso
def implementationExcludes = project.ext.implementationExcludes
dependencies{
    // 处理所有的 xxximplementation 依赖
    implementationDependencies.each { k, v -> implementation v }   
    // 处理 annotationProcessor 依赖
    processors.each { k, v -> annotationProcessor v }
    // 处理所有包含 exclude 的依赖
    implementationExcludes.each { entry ->
        implementation(entry.key) {
            entry.value.each { childEntry ->
                exclude(group: childEntry)
            }
        }
    }

这样做的优点在于
1.后续添加依赖不需要改动build.gradle,直接在config.gradle中添加即可
2.精简了build.gradle的长度

但是还是有一些痛点:

1.不支持代码提示
2.不支持单击跳转
3.多模块开发时,不同模块相同的依赖需要复制粘贴。

使用buildSrc管理Gradle依赖

创建一个名为buildSrc的module.因为运行Gradle时,它会检查项目中是否存在一个名为buildSrc的目录。然后Gradle会自动编译并测试这段代码,并将其放入构建脚本的类路径中。您不需要提供任何进一步的操作提示。 所以也不需要在settings.gradle文件中添加include xxx。

步骤如下:

1.新建文件 build.gradle.kts,并同步 gradle

plugins {
    `kotlin-dsl`
}
repositories {
    jcenter()
}

2.创建 /src/main/java/Version.kt 文件,在 Dependencies.kt 文件中编写依赖管理代码:

object BuildVersion {
    const val compileSdk = 30
    const val buildTools = "30.0.2"
    const val minSdk = 21
    const val targetSdk = 30
    const val versionCode = 1
    const val versionName = "1.0.0"
}

object Versions {
    const val kotlin = "1.4.30"
    const val core_ktx = "1.3.2"
    const val appcompat = "1.2.0"
    const val material = "1.3.0"
    const val retrofit = "2.9.0"
}

object Libs {
    const val kotlin_stdlib = "org.jetbrains.kotlin:kotlin-stdlib:${Versions.kotlin}"
    const val core_ktx = "androidx.core:core-ktx:1.3.2:${Versions.core_ktx}"
    const val appcompat = "androidx.appcompat:appcompat:${Versions.appcompat}"
    const val material = "com.google.android.material:material:${Versions.material}"
    const val retrofit = "com.squareup.retrofit2:retrofit:${Versions.retrofit}"
}

3.在 module 中 build.gradle 应用依赖项,如下

plugins {
    id 'com.android.library'
    id 'kotlin-android'
}

android {
    compileSdkVersion BuildVersion.compileSdk
    buildToolsVersion BuildVersion.buildTools

    defaultConfig {
        minSdkVersion BuildVersion.minSdk
        targetSdkVersion BuildVersion.targetSdk
        versionCode BuildVersion.versionCode
        versionName BuildVersion.versionName
        ...
    }

    ...
}

dependencies {

    implementation Libs.kotlin_stdlib
    implementation Libs.core_ktx
    implementation Libs.appcompat
    implementation Libs.material

    implementation Libs.retrofit
}

缺点:由于 buildSrc 是对全局的所有 module 的配置,因此在构建速度上会慢一些。

使用includeBuild统一配置依赖版本

步骤如下:

1.先创建一个java或kotlin的library.

2.然后在创建好的library的build.gradle文件中添加以下代码,我这里用的kotlin,如果用的Java,则 apply plugin: 'java-library'。

apply plugin: 'kotlin'
apply plugin: 'java-gradle-plugin'

buildscript {
    repositories {
        jcenter()
        google()
    }
    dependencies {
        classpath "org.jetbrains.kotlin:kotlin-gradle-plugin:1.4.10"

    }
}

repositories {
    jcenter()
    google()
}

dependencies {
    implementation gradleApi()
    implementation "org.jetbrains.kotlin:kotlin-gradle-plugin:1.4.10"
}

gradlePlugin{
    plugins{
        version{
            //id 对应package name
            id = "com.example.version"
            //implementationClass 则是对应我们第三步需要创建的一个类,其内容为包名 + 类名
            implementationClass = "com.example.version.DependencyVersionPlugin"
        }
    }
}

3.在com.example.version路径下创建一个类

class DependencyVersionPlugin : Plugin<Project> {
    override fun apply(target: Project) {

    }
}

4.在src/main/java的路径下添加文件,文件里就是所有的依赖集合

5.在settings.gradle添加

//括号里对应library名
includeBuild("version")

6.然后在需要的module,添加

plugins{
    id  "com.example.version"
}

sync之后,就可使用了。

dependencies {
    implementation fileTree(dir: 'libs', include: ['*.jar'])
    implementation(AndroidX.coreKtx)
    implementation(AndroidX.appcompat)
    implementation(AndroidX.constraintlayout)
}

这里有一个坑,不能将

android {
    compileSdkVersion ProjectBuildConfigs.compileSdkVersion
    buildToolsVersion ProjectBuildConfigs.buildToolsVersion

    defaultConfig {
        minSdkVersion ProjectBuildConfigs.minSdkVersion
        targetSdkVersion ProjectBuildConfigs.targetSdkVersion
        versionCode ProjectBuildConfigs.versionCode
        versionName ProjectBuildConfigs.versionName

        consumerProguardFiles "consumer-rules.pro"

        ndk {
            // 设置支持的SO库架构
            //abiFilters 'armeabi', 'x86', 'armeabi-v7a', 'x86_64', 'arm64-v8a'
            abiFilters 'armeabi-v7a', 'arm64-v8a', 'x86'
        }
    }
}

这些公共的东西放在一个公共的base_xxx.gradle,然后再在build.gradle里面 apply from "base_xxx.gradle ",这样会一直报compileSdkVersion is not specified.的错误。我在这上面弄了好久,一直怀疑是写的有问题。


b7841a723bad5c2eb56287073db3324.png

使用红色框的方式会一直报错,所以我们需要采用绿色框的方式

还有一个坑就是不能套娃:

object Kotlin {
        const val Kotlin = "org.jetbrains.kotlin:kotlin-stdlib:${Version.Kotlin}"
        const val CoroutinesCore =
            "org.jetbrains.kotlinx:kotlinx-coroutines-core:${Version.Coroutines}"
        const val CoroutinesAndroid =
            "org.jetbrains.kotlinx:kotlinx-coroutines-android:${Version.Coroutines}"
        object GitHub {
                  const val OkHttp = "com.squareup.okhttp3:okhttp:${Version.OkHttp}"
                  const val OkHttpInterceptorLogging =
            "com.squareup.okhttp3:logging-interceptor:${Version.OkHttpInterceptorLogging}"
                  const val Retrofit = "com.squareup.retrofit2:retrofit:${Version.Retrofit}"
            }
    }

这样会编译不过。

使用Version Catalog统一配置依赖版本

Catalog 是Gradle7.0推出了一个新的特性,它支持以下特性:

1.对所有module可见,可统一管理所有module的依赖

2.支持声明依赖bundles,即总是一起使用的依赖可以组合在一起

3.支持版本号与依赖名分离,可以在多个依赖间共享版本号

4.支持在单独的libs.versions.toml文件中配置依赖

5.支持在项目间共享依赖

具体使用请看【Gradle7.0】依赖统一管理的全新方式,了解一下~

参考资料

【Gradle7.0】依赖统一管理的全新方式,了解一下~

git https://github.com/fengyuehan/Test/tree/master/version

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,684评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,143评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,214评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,788评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,796评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,665评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,027评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,679评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,346评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,664评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,766评论 1 331
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,412评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,015评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,974评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,073评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,501评论 2 343

推荐阅读更多精彩内容