Gradle 自动化项目构建-Gradle 核心之 Project

一、前言


从明面上看,Gradle 是一款强大的构建工具,但 Gradle 不仅仅是一款强大的构建工具,它更像是一个编程框架。Gradle 的组成可以细分为如下三个方面:

  • groovy 核心语法:包括 groovy 基本语法、闭包、数据结构、面向对象等等。
  • Android DSL(build scrpit block):Android 插件在 Gradle 所特有的东西,我们可以在不同的 build scrpit block 中去做不同的事情。
  • Gradle API:包含 Project、Task、Setting 等等。

可以看到,Gradle 的语法是以 groovy 为基础的,而且它还有自己独有的 API,所以我们可以把 Gradle 认作是一款编程框架,利用 Gradle 我们可以在编程中去实现项目构建过程中的所有需求。想要随心所欲地使用 Gradle,我们必须提前掌握好 groovy。需要注意的是,Groovy 是一门语言,而 DSL 一种特定领域的配置文件,Gradle 是基于 Groovy 的一种框架工具,而 gradlew 则是 gradle 的一个兼容包装工具。

Gradle 有以下优势:

  1. 灵活性:相对于 Maven、Ant 等构建工具,Gradle 提供了一系列的 API 让我们有能力去修改或定制项目的构建过程。例如我们可以利用 Gradle 去动态修改生成的 APK 包名。
  2. 粒度性:使用 Maven、Ant 等构建工具时,我们的源代码和构建脚本是独立的,而且我们也不知道其内部的处理是怎样的。但是 Gradle 则不同,它从源代码的编译、资源的编译、到生成 APK 的过程中都是一个接一个来执行的。此外,Gradle 构建的粒度细化到了每一个 task 之中。并且它所有的 Task 源码都是开源的,在我们掌握了这一整套打包流程后,我们就可以通过修改它的 Task 去动态改变其执行流程。例如 Tinker 框架的实现过程中,它通过动态地修改 Gradle 的打包过程生成 APK 的同时,也生成了各种补丁文件。
  3. 扩展性:Gradle 支持插件机制,所以我们可以复用这些插件,就如同复用库一样简单方便。
  4. 兼容性:Gradle 不仅自身功能强大,而且它还能兼容所有的 Maven、Ant 功能,也就是说,Gradle 吸取了所有构建工具的长处。

可以看到,Gradle 相比于其它构建工具,其好处不言而喻,而其最核心的原因就是因为 Gradle 是一套编程框架。

二、Gradle 的生命周期


所谓 Gradle 的生命周期,即 gradle 的执行流程,也就是 Gradle 先执行什么后执行什么。我们看下它的流程图:

可以看到,gradle 的执行流程分了 初始化、配置、执行 三个阶段,上图中的 project、task 我们接下来几篇会详细介绍。下面我们看看这几个阶段。

 2.1、初始化阶段

初始化阶段会读取根工程中的 setting.gradle 中的 include 信息,确定有多少工程加入构建,然后会为每一个项目(build.gradle 脚本文件)创建一个个与之对应的 Project 实例,最终形成一个项目的层次结构。

与初始化阶段相关的脚本文件是 settings.gradle,而一个 settings.gradle 脚本对应一个 Settings 对象,我们最常用来声明项目的层次结构的 include 就是 Settings 对象下的一个方法,在 Gradle 初始化的时候会构造一个 Settings 实例对象,以执行各个 Project 的初始化配置。

此外,在 settings.gradle 文件中,我们可以指定其它 project 的位置,这样就可以将其它外部工程中的 moudle 导入到当前的工程之中了。示例代码如下所示:

if (useSpeechMoudle) {// 导入其它 App 的 speech 语音模块include "speech"project(":speech").projectDir = new File("../OtherApp/speech")
}

2.2、配置阶段

配置阶段的任务是执行各项目下的 build.gradle 脚本,完成 Project 的配置,与此同时,会构造 Task 任务依赖关系图以便在执行阶段按照依赖关系执行 Task。而在配置阶段执行的代码通常来说都会包括以下三个部分的内容,如下所示:

  • 1)、build.gralde 中的各种语句。
  • 2)、闭包。
  • 3)、Task 中的配置段语句。

需要注意的是,执行任何 Gradle 命令,在初始化阶段和配置阶段的代码都会被执行。

2.3、执行阶段

在配置阶段结束后,Gradle 会根据各个任务 Task 的依赖关系来创建一个有向无环图,我们可以通过 Gradle 对象的 getTaskGraph 方法来得到该有向无环图。并且当有向无环图构建完成之后,所有 Task 执行之前,我们可以通过 whenReady(groovy.lang.Closure) 或者 addTaskExecutionGraphListener(TaskExecutionGraphListener) 来接收相应的通知,其代码如下所示:

gradle.getTaskGraph().addTaskExecutionGraphListener(new TaskExecutionGraphListener() {@Overridevoid graphPopulated(TaskExecutionGraph graph) {}
})

然后,Gradle 构建系统会通过调用 gradle <任务名> 来执行相应的各个任务。

可以看到,整个 Gradle 生命周期的流程包含如下 四个部分:

  • 首先,解析 settings.gradle 来获取模块信息,这是初始化阶段。
  • 然后,配置每个模块,配置的时候并不会执行 task。
  • 接着,配置完了以后,有一个重要的回调 project.afterEvaluate,它表示所有的模块都已经配置完了,可以准备执行 task 了。
  • 最后,执行指定的 task 及其依赖的 task。

在 Gradle 构建命令中,最为复杂的命令可以说是 gradle build 这个命令了,因为项目的构建过程中需要依赖很多其它的 task。这里,我们以 Java 项目的构建过程看看它所依赖的 tasks 及其组成的有向无环图,如下所示:

2.4、生命周期监听

上面我们学习了 Gradle 的执行生命流程,下面我们在它的监听回调中做一些输出。

首先在项目根目录的 build.gradle 中添加如下监听代码:

在根目录的 setting.gradle 中添加如下代码:

接下来我们执行一个简单的 gradle 命令:gradle clean


Gradle 核心之 Project

一、前言


Project 是 Gradle 构建整个应用程序的入口,所以它非常重要。我们看下面这张图:

上图是我创建的一个 Android 工程,并添加了一个 test module。我们在命令行中输入 gradle projects 命令看看有哪些 project:

可以看到输出了三个 project,其中 GradleTextProject 是根 project,而 app、test 是子 project。根 project 的作用是管理所有的 子 project。准确来说有 build.gradle 文件的目录即是 project。一个子 project 对应一个输出,具体输出什么由 build.gradle 配置去决定。

二、project 核心 api


 在 Project 中有很多的 API,但是根据它们的属性和用途我们可以将其分解为六大部分,如下图所示:

对于 Project 中各个部分的作用,我们可以先来大致了解下,以便为 Project 的 API 体系建立一个整体的感知能力,如下所示:

  1. Project 相关 API:让当前 Project 拥有了操作它的父 Project 以及管理它的子 Project 的能力。
  2. Task 相关 API:为当前 Project 提供了新增 Task 以及管理已有 Task 的能力。由于 task 非常重要,我们将在下一篇进行讲解。
  3. 属性相关的 Api:Gradle 会预先为我们提供一些 Project 属性,而属性相关的 api 让我们拥有了为 Project 添加额外属性的能力。
  4. File 相关 Api:Project File 相关的 API 主要用来操作我们当前 Project 下的一些文件处理。
  5. Gradle 生命周期 API:即我们在上一篇讲解过的 Gradle 核心之生命周期。
  6. 其它 API:添加依赖、添加配置、引入外部文件等等零散 API 的聚合。

2.1、Project 相关 API

通过 gradle 管理的工程都会有一个根工程 project,根工程用来管理子工程。下面我们来看看 Project 相关的 API。

2.1.1 getAllprojects()

getAllprojects 表示获取所有 project 的实例,示例代码如下所示:

我们调用了 getAllprojects 方法返回一个包含根 project 与其子 project 的 Set 集合,并链式调用了 eachWithIndex 遍历 Set 集合。接着,我们会判断当前的下标 index 是否是0,如果是则表明当前遍历的是 rootProject,则输出 rootProject 的名字,否则输出 child project 的名字。

然后我们在命令行执行 gradle clean,其运行结果可以看到,会先配置我们的 rootProject,并输出了对应的工程信息。接着便会执行子工程 app 的配置。rootProject 与其旗下的各个子工程组成了一个树形结构。

2.1.2 getSubprojects()

getSubprojects 表示获取当前工程下所有子工程的实例,示例代码如下所示:

同 getAllprojects 的用法一样,getSubprojects 方法返回了一个包含子 project 的 Set 集合。

2.1.3 getParent()

getParent() 是获取父 project 的方法。

 可以看到,执行 gradle clean 后输出了 test module 这个 project 的父 project,也就是 GradleTextProject。需要注意的是,如果在根目录的 build.gradle 中调用 getParent() ,由于根 project 没有父节点了,所有返回的是 null。

2.1.4 getRootProject()

getRootProject() 获取的是根节点 project。

形成的 project 树中肯定是有根节点的,所以在任意子节点 project 中调用 getRootProject 都返回的是根节点 project,所以肯定不会返回空。

2.1.5 project()

project 表示的是指定工程的实例,然后在闭包中对其进行操作。可以看到,在 project 方法中两个参数,一个是指定工程的路径,另一个是用来配置该工程的闭包。下面我们看看如何灵活地使用 project,示例代码如下所示: 

2.1.6 allprojects()

allprojects 表示用于配置当前 project 及其每一个子 project,在 allprojects 中我们一般用来配置一些通用的配置,比如最常见的全局仓库配置。如下所示:

当我们用熟练后,可以省略闭包的参数:

2.1.7 subprojects()

subprojects 用于统一配置当前 project 下的所有子 project, 给所有的子工程引 将 aar 文件上传置 Maven 服务器的配置脚本,示例代码如下所示:

 
  1. subprojects {

  2. if (project.plugins.hasPlugin("com.android.library")) {

  3. apply from: '../publishToMaven.gradle'

  4. }

  5. }

在上述示例代码中,我们会先判断当前 project 下的子 project 是不是库,如果是库才有必要引入 publishToMaven 脚本。需要注意与 allprojects() 的区别是,subprojects() 不包含当前 project。

2.2 属性相关API

Project 提供了默认的 7 个属性,我们先来看看这些属性:

第一个属性 DEFAULT_BUILD_FILE = "build.gradle" 表明默认读取的配置文件是 build.gradle,这也证明了上面说有 build.gradle 的文件夹就是一个 project 的结论。

第二个属性 PATH_SEPARATOR 表示的是分隔符。

第三个属性 DEFAULT_BUILD_DIR_NAME 表示默认的输出文件夹,每个工程都会有一个 build 文件夹存放 project 输出。

后面几个属性不常用到,就不详细说明了。这么少的属性显然无法满足我们各种各样的构建需求,gradle 为我们提供了一种去扩展 project 属性的方式。主要有两种扩展方式,下面我们来看看。

2.2.1 ext 扩展属性

我们可以使用 ext 扩展属性修改默认情况下 app 或其他 module 的 build.gradle 配置,如下所示:

project 中 ext 加闭包即定义扩展属性,我们可以在每个 project 的 build.gradle 文件中定义 ext 扩展属性,但当我们有多个 project 的时候这种写法也很麻烦。这时候我们可以把 ext 放到上一节我们学习的 allprojects()、subprojects() 中,然后在子 project 中用 this 关键字引用即可。

 另外,我们也可以去掉 subprojects(),在根目录中直接设置 ext,然后在子 project 中通过 this.rootProject 来引用。

另外也可以直接通过 this 来直接使用,因为子 project 是继承父 project 的,所以父 project 中定义的属性,子 project 可以直接使用。随着版本的迭代,演变出了最优方案:将扩展属性单独定义到一个新的 gradle 文件中,这样就可以减少根工程的配置代码,同时也更模块化了我们的变量。这里我们通常会将其命名为 config.gradle,如下:

可以看到,在 config.gradle 中分了各个区块,在每个区块中定义了一个 map,在 map 中定义各种 key、value。然后在我们的根目录下的 build.gradle 中引入这个 gradle:

引用完成后就可以在我们的子 project 的 build.gradle 中按区块引用即可:

2.2.2 gradle.properties 里定义扩展属性

除了使用 ext 扩展属性定义额外的属性之外,我们也可以在 gradle.properties 下定义扩展属性,其示例代码如下所示:

 
  1. // 在 gradle.properties 中

  2. mCompileVersion = 27

  3. // 在 app moudle 下的 build.gradle 中

  4. compileSdkVersion mCompileVersion.toInteger()

2.3 文件相关API

2.3.1 路径获取相关API

关于路径获取的 API 常用的有三种,其示例代码如下所示:

2.3.2 文件操作API

groovy 中的文件操作 API 可以用在 project 中,而本节讲解的 project 中文件操作 API 可以在 project 下去更方便的对文件进行操作。下面我们来看看有哪些操作。

文件定位:常用的文件定位 API 有下面两个方法:

 
  1. //定位单个文件

  2. File file(Object path);

  3. //定位多个文件

  4. ConfigurableFileCollection files(Object... paths);

使用如下所示:

文件拷贝:常用的文件拷贝 API 为 copy,不仅可以对文件进行拷贝,也可以对文件夹进行拷贝。其示例代码如下所示:

在实际项目中的使用一般如下:

 
  1. tasks.whenTaskAdded { task ->

  2. if (task.name.equalsIgnoreCase("assembleRelease")) {

  3. // 如果是assembleRelease任务,在最后执行导出apk以及mapping目录到指定目录

  4. task.doLast {

  5. outputReleaseFile()

  6. }

  7. }

  8. }

  9. void outputReleaseFile() {

  10. android.applicationVariants.all { variant ->

  11. // 如果是正式版打包

  12. if (variant.name.equalsIgnoreCase("release")) {

  13. File outputPath = new File("$rootDir" + File.separator + "release_app" + File.separator

  14. + android.defaultConfig.versionName)

  15. println(String.format('拷贝生成文件到指定目录[%s]', outputPath.getAbsolutePath()))

  16. // 拷贝apk文件

  17. copy {

  18. from variant.outputs[0].outputFile

  19. into outputPath

  20. // 重命名导出名称

  21. rename {

  22. 'account_system' + variant.name + '_' + android.defaultConfig.versionName + ".apk"

  23. }

  24. }

  25. // 拷贝mapping目录

  26. copy {

  27. from variant.mappingFile.getParentFile()

  28. into new File(outputPath, 'mapping')

  29. }

  30. }

  31. }

  32. }

变体其实就是我们的 apk,变体我们后面再介绍。

2.3.3 文件树的遍历

我们可以 使用 fileTree 将当前目录转换为文件数的形式,然后便可以获取到每一个树元素(节点)进行相应的操作,其示例代码如下所示:

2.4 其他API

其他API包含两部分:

2.4.1 依赖相关API

根项目下的 buildscript 用于配置项目核心的依赖,使用如下:

当我们熟练使用闭包后可以简写如下:

需要注意的是不同于根项目 buildscript 中的 dependencies 是用来配置我们 Gradle 工程的插件依赖的,而 app moudle 下的 dependencies 是用来为应用程序添加第三方依赖的。关于 app moudle 下的依赖使用这里我们需要注意下 exclude 与 transitive 的使用 即可,示例代码如下所示:

 
  1. implementation(rootProject.ext.dependencies.glide) {        

  2. // 排除依赖:一般用于解决资源、代码冲突相关的问题        

  3. exclude module: 'support-v4'         

  4. // 传递依赖:A => B => C ,B 中使用到了 C 中的依赖,且 A 依赖于 B,如果打开传递依赖,则 A 能使用到 B 中所使用的 C 中的依赖

  5. //默认都是不打开,即 false 禁止传递依赖     

  6. transitive false 

  7. }

 传递依赖文字描述有点抽象,我们来看下面这张图就可以明白了:

2.4.2 外部命令执行

如果 Gradle 的 API 能满足我们的需求时尽量使用 Gradle API,不行的化我们就可以考虑使用 Gradle 提供的 exec 来执行外部命令,下面我们就使用 exec 命令来 将当前工程下新生产的 APK 文件拷贝到电脑下的 Downloads 目录中,示例代码如下所示:

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/357566.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

抖音多功能全自动引流工具,支持评论关注私信留痕点赞等,让你的抖音粉丝暴涨!

随着短视频行业的火爆&#xff0c;越来越多的人开始关注抖音这个平台。然而&#xff0c;如何在抖音上获得更多的关注和粉丝&#xff0c;成为了许多人面临的难题。为了帮助大家解决这个问题&#xff0c;今天我们将为大家推荐一款抖音多功能全自动引流脚本&#xff0c;这款脚本可…

C | 在ubuntu22下开发的一些配置

目录 VScode设置 要下载的插件&#xff1a; 卸载VScode的话就是哪装的哪删。 浅用gcc 预处理指令 使用gcc 语言编译过程 1. 预处理&#xff08;Preprocessing&#xff09; 2. 编译&#xff08;Compilation&#xff09; 3. 汇编&#xff08;Assembly&#xff09; 4. …

rockchip linux sdk指定编译配置文件

SDK&#xff1a;rk3568_linux4.19_V1.4.0 硬件平台&#xff1a;RK3566 一、指定板级配置文件 板级配置文件在<SDK>/device/rockchip/rk3566_rk3568目录下。 1、方法1 ./build.sh后⾯加上板级配置⽂件&#xff0c;例如&#xff1a; ./build.sh /device/rockchip/rk3…

setInterval 定时任务执行时间不准验证

一般在处理定时任务的时候都使用setInterval间隔定时调用任务。 setInterval(() > {console.log("interval"); }, 2 * 1000);我们定义的是两秒执行一次&#xff0c;但是浏览器实际执行的间隔时间只多不少。这是由于浏览器执行 JS 是单线程模式&#xff0c;使用se…

八、(正点原子)Linux内核定时器实验

定时器是我们最常用到的功能&#xff0c;一般用来完成定时功能&#xff0c;本章我们就来学习一下 Linux 内核提供的定时器 API 函数&#xff0c;通过这些定时器 API 函数我们可以完成很多要求定时的应用。 Linux内核也提供了短延时函数&#xff0c;比如微秒、纳秒、毫秒延时函数…

【电路笔记】-共发射极放大器

共发射极放大器 文章目录 共发射极放大器1、概述2、完整的CEA配置3、直流等效电路4、交流等效电路5、输入阻抗6、输出阻抗7、电压增益8、微分电容的重要性9、信号源的衰减10、电流增益11、相位反转12、总结1、概述 在本文中,我们将介绍基于双极晶体管的放大器的最后一种拓扑:…

Piecewise Jerk Speed 论文以及代码解析

目录 1 算法原理 1.1 优化模型离散方式 1.1.1 Temporal Parameter Discretization 1.2 优化问题建模 1.2.1 Cost function 1.2.2 Constraints 2 ST-Graph 3 代码实现 ​编辑 3.1 STBoundsDecider 1 算法原理 1.1 优化模型离散方式 在处理最优化问题时&#xff0c;…

ardupilot开发 --- RTSP视频流 篇

我年轻时很穷&#xff0c;努力了几年&#xff0c;终于不再年轻了 0. 一些概念1. Ubuntu搭建RTSP服务器的方式2. 在Ubuntu上搭建RTSP服务器3. 推流4. 拉流、播放5. 借鉴的一些例子6. 其他参考文献 0. 一些概念 RTSP服务、RTSP推流、RTSP拉流&#xff0c;缺一不可&#xff0c;尤其…

Web实时通信的学习之旅:WebSocket入门指南及示例演示

文章目录 WebSocket的特点1、工作原理2、特点3、WebSocket 协议介绍4、安全性 WebSocket的使用一、服务端1、创建实例&#xff1a;创建一个webScoket实例对象1.1、WebSocket.Server(options[&#xff0c;callback])方法中options对象所支持的参数1.2、同样也有一个加密的 wss:/…

mysql中in参数过多该如何优化

优化方式概述 未优化前 SELECT * FROM rb_product rb where sku in(1022044,1009786)方案2示例 public static void main(String[] args) {//往list里面设置3000个值List<String> list new ArrayList<>();for (int i 0; i < 3000; i) {list.add(""…

RNN循环卷积神经网络

1.定义 RNN &#xff08;Recurrent Neural Network&#xff0c;RNN&#xff09;循环卷积神经网络&#xff0c;用于处理序列数据。 序列数据&#xff1a;按照一定的顺序排列的数据&#xff0c;可以是时间顺序、空间顺序、逻辑顺序。 eg:电影、语言 2.特点 传统神经网络模型无法…

Java开发-面试题-0007-GPT和MBR的区别

Java开发-面试题-0007-GPT和MBR的区别 更多内容欢迎关注我&#xff08;持续更新中&#xff0c;欢迎Star✨&#xff09; Github&#xff1a;CodeZeng1998/Java-Developer-Work-Note 技术公众号&#xff1a;CodeZeng1998&#xff08;纯纯技术文&#xff09; 生活公众号&#…

[Shell编程学习路线]——While循环应用技巧 (详细讲解)

&#x1f3e1;作者主页&#xff1a;点击&#xff01; &#x1f6e0;️Shell编程专栏&#xff1a;点击&#xff01; ⏰️创作时间&#xff1a;2024年6月20日16点30分 &#x1f004;️文章质量&#xff1a;95分 目录 ————前言———— 基本结构 图示原理 示例 测试 …

Go语言的诞生背景

人不走空 &#x1f308;个人主页&#xff1a;人不走空 &#x1f496;系列专栏&#xff1a;算法专题 ⏰诗词歌赋&#xff1a;斯是陋室&#xff0c;惟吾德馨 目录 &#x1f308;个人主页&#xff1a;人不走空 &#x1f496;系列专栏&#xff1a;算法专题 ⏰诗词歌…

Spark Core内核调度机制详解(第5天)

系列文章目录 如何构建DAG执行流程图 (掌握)如何划分Stage阶段 (掌握)Driver底层是如何运转 (掌握)确定需要构建多少分区(线程) (掌握) 文章目录 系列文章目录引言一、Spark内核调度&#xff08;掌握&#xff09;1.1、内容概述1.2、RDD的依赖1.3、DAG和Stage1.4、Spark Shuffl…

云计算 私有云openstack 搭建指导 【99%成功率】

本人已经搭了很多次&#xff0c;也失败了很多次&#xff0c;这里面有很多失败的经验&#xff0c;以及提示&#xff0c;请按提示搭建&#xff0c;相信你也可以的&#xff01; 不积小流无以成江海&#xff0c;多尝试才能吸取经验 开始搭建openstack 准备文件&#xff08;如果没有…

练手代码之使用Python实现合并PDF文件

如果你有合并PDF的需要&#xff0c;你会怎么办 我们无所不能的程序员会选择写一个Python代码来实现&#xff08;谁会这么无聊&#xff1f;是我&#xff09;&#xff0c;如果真的有PDF操作需要&#xff0c;我推荐你使用PDF Expert这个软件哈~ 话不多说直接上代码&#xff1a; …

Pip换源秘籍:让你的Python包飞行起来!

在Python的包管理中&#xff0c;Pip是最重要的工具之一。它允许开发者从Python Package Index (PyPI)安装包&#xff0c;但有时由于网络问题或服务器负载过高&#xff0c;直接从PyPI安装包可能会非常慢。这时&#xff0c;更换Pip源到一个更快的镜像站点是一个常见的解决方案。本…

版本控制工具-git的基本使用

目录 前言一、git简介二、git工作流程三、git基本命令3.1 创建本地仓库3.2 将工作区内容提交到本地仓库3.3 将本地仓库内容推送到远程仓库 前言 本篇文章介绍git的一般工作流程 一、git简介 Git是一个开源的分布式版本控制软件&#xff0c;常用于项目的版本管理 Git是Linux …

破碎的像素地牢探险:游戏分享

软件介绍 《破碎的像素地牢》是开源一款地牢冒险探索类的游戏&#xff0c;融合了日系RPG经典风格&#xff0c;玩家将控制主角进行未知场景的探索。除了经典地牢玩法外&#xff0c;游戏还添加了更多创意内容&#xff0c;如黑屏状态前的挑战性等&#xff0c;使得游戏更加富有挑战…