parent
f6a1cd73a9
commit
ab136c224e
@ -0,0 +1,431 @@ |
|||||||
|
--- |
||||||
|
一篇文章读完《Android Gradle 权威指南》 |
||||||
|
--- |
||||||
|
|
||||||
|
#### 目录 |
||||||
|
|
||||||
|
1. 思维导图 |
||||||
|
2. 前言 |
||||||
|
3. Gradle 基础知识 |
||||||
|
- Gradle 命令行 |
||||||
|
- Gradle 日志 |
||||||
|
- Groovy 基础 |
||||||
|
4. Gradle 构建脚本基础 |
||||||
|
- Settings 和 Build 文件 |
||||||
|
- 创建任务和任务依赖 |
||||||
|
- 自定义属性 |
||||||
|
5. Gradle 任务进阶 |
||||||
|
- 任务分组和描述 |
||||||
|
- 任务执行分析 |
||||||
|
- 任务的 OnlyIf 断言 |
||||||
|
|
||||||
|
#### 思维导图 |
||||||
|
|
||||||
|
#### 前言 |
||||||
|
|
||||||
|
在写完 [Gradle Plugin 入门指南](https://github.com/Omooo/Android-Notes/blob/master/blogs/Android/Gradle/Gradle Plugin.md) 和 [Gradle Plugin 实践之 TinyPng Plugin](https://github.com/Omooo/Android-Notes/blob/master/blogs/Android/Gradle/Gralde%20Plugin%20%E5%AE%9E%E8%B7%B5%E4%B9%8B%20TinyPng%20Plugin.md) 之后,总会有一些疑问,如果你仔细阅读过这两篇文章,你甚至会发现文章还有一些错误的说法,比如以下命令中 -q 参数的含义: |
||||||
|
|
||||||
|
``` |
||||||
|
./gradlew task -q build |
||||||
|
``` |
||||||
|
|
||||||
|
-q 的含义并不是表示静默输出,而是…..(下面会讲到 |
||||||
|
|
||||||
|
《Android Gradle 权威指南》这本书理解起来相对比较简单,但是它解决了我的很多疑惑,我会把书中重要知识都在本文中记录下来。我希望读书是一种边解惑边吸收新知识的过程,我也正是以这种方式来写这篇文章的。 |
||||||
|
|
||||||
|
感谢作者的著作~,受益匪浅。 |
||||||
|
|
||||||
|
#### Gradle 基础知识 |
||||||
|
|
||||||
|
Gradle 基础知识分为三块:Gradle 命令行、Gradle 日志、Groovy 基础。 |
||||||
|
|
||||||
|
##### Gradle 命令行 |
||||||
|
|
||||||
|
Gradle 命令一般用于执行一些 Task 。 |
||||||
|
|
||||||
|
在前两篇文章,我们也接触到了两个 Gradle 命令,一个是用于执行我们自定义 task 的命令: |
||||||
|
|
||||||
|
``` |
||||||
|
./gradlew task myCustomTask |
||||||
|
``` |
||||||
|
|
||||||
|
一个是用于执行上传 jar 包依赖的命令: |
||||||
|
|
||||||
|
``` |
||||||
|
./gradlew task uploadArchives |
||||||
|
``` |
||||||
|
|
||||||
|
这个 uploadArchives Task 其实是 java plugin 里面的 Task,也就是说我们依赖了这个插件: |
||||||
|
|
||||||
|
```java |
||||||
|
apply plugin: 'java' |
||||||
|
``` |
||||||
|
|
||||||
|
起初我一直以为这个 task 是 android 里面的,在 app 模块中我们会依赖 android plugin: |
||||||
|
|
||||||
|
```java |
||||||
|
apply plugin: 'com.android.application' |
||||||
|
``` |
||||||
|
|
||||||
|
之所以,我们在 app 模块也能使用 uploadArchives 是因为 android plugin 是继承 java plugin 的。 |
||||||
|
|
||||||
|
除此之外,Android 还内置了很多有用的 Task,我们可以通过执行: |
||||||
|
|
||||||
|
``` |
||||||
|
./gradlew tasks |
||||||
|
``` |
||||||
|
|
||||||
|
来查看所有的 tasks,这里我就贴出了部分输出: |
||||||
|
|
||||||
|
``` |
||||||
|
> Task :tasks |
||||||
|
|
||||||
|
------------------------------------------------------------ |
||||||
|
All tasks runnable from root project |
||||||
|
------------------------------------------------------------ |
||||||
|
|
||||||
|
Android tasks |
||||||
|
------------- |
||||||
|
androidDependencies - Displays the Android dependencies of the project. |
||||||
|
signingReport - Displays the signing info for the base and test modules |
||||||
|
sourceSets - Prints out all the source sets defined in this project. |
||||||
|
|
||||||
|
Build tasks |
||||||
|
----------- |
||||||
|
//... |
||||||
|
build - Assembles and tests this project. |
||||||
|
clean - Deletes the build directory. |
||||||
|
//... |
||||||
|
``` |
||||||
|
|
||||||
|
可以看到,tasks 是以分组进行输出的,前面是 task name,后面是 task description,我们自定义的 task 也是可以添加到某个分组里面的,也可以新建一个分组然后添加。 |
||||||
|
|
||||||
|
比如我们可以执行上面 Android tasks 中的 androidDependencies 来查看项目中的所有依赖: |
||||||
|
|
||||||
|
``` |
||||||
|
./gradlew task androidDependencies |
||||||
|
``` |
||||||
|
|
||||||
|
当然,我们也可以一次执行多个任务,比如: |
||||||
|
|
||||||
|
``` |
||||||
|
./gradlew task clean build |
||||||
|
``` |
||||||
|
|
||||||
|
但是,需要主要的是两者的依赖关系,这个我们后面自定义 Task 的时候会讲到。 |
||||||
|
|
||||||
|
##### Gradle 日志 |
||||||
|
|
||||||
|
这个就是解决我们开头的那个错误的说法了。 |
||||||
|
|
||||||
|
Gradle 日志是有日志级别的: |
||||||
|
|
||||||
|
| 级别 | 用于 | |
||||||
|
| --------- | -------- | |
||||||
|
| ERROR | 错误消息 | |
||||||
|
| QUIET | 重要消息 | |
||||||
|
| WARNING | 警告消息 | |
||||||
|
| LIFECYCLE | 进度消息 | |
||||||
|
| INFO | 信息消息 | |
||||||
|
| DEBUG | 调试信息 | |
||||||
|
|
||||||
|
我们指定 -q 的日志级别并不是说输出重要信息,而是输出 QUIET 及其更高级别的日志信息,总结下就是: |
||||||
|
|
||||||
|
| 参数 | 说明 | |
||||||
|
| --------------- | ---------------------- | |
||||||
|
| 无参 | LIFECYCLE 及其更高级别 | |
||||||
|
| -q 或者 --quiet | QUIET 及其更高级别 | |
||||||
|
| -i 或者 --info | INFO 及其更高级别 | |
||||||
|
| -d 或者 --debug | DEBUG 及其更高级别 | |
||||||
|
|
||||||
|
我们在写 Gradle Plugin 是时候,验证是否执行成功是通过打印一句话来测试的: |
||||||
|
|
||||||
|
```groovy |
||||||
|
println 'Gralde Task 执行完毕~' |
||||||
|
``` |
||||||
|
|
||||||
|
使用 print 系列方法,把日志信息输出到标准的控制台输出流,它被 Gradle 定向为 QUIET 级别日志,但是 Gradle 是有日志 API 的: |
||||||
|
|
||||||
|
```groovy |
||||||
|
logger.quiet("Quiet 信息") |
||||||
|
logger.error("Error 信息") |
||||||
|
logger.warn("Warn 信息") |
||||||
|
logger.lifecycle("Lifecycle 信息") |
||||||
|
logger.info('Info 信息') |
||||||
|
logger.debug("Debug 信息") |
||||||
|
``` |
||||||
|
|
||||||
|
这里其实是调用的 Project 的 getLogger() 方法获取的 Logger 对象的实例。 |
||||||
|
|
||||||
|
##### Groovy 基础 |
||||||
|
|
||||||
|
这一小节我并不会介绍过多的 Groovy 基础知识,因为完全没必要,大家看文档就好了。这里只说一点,那就是方法的调用。 |
||||||
|
|
||||||
|
```groovy |
||||||
|
//定义方法 |
||||||
|
def show(int a, int b) { |
||||||
|
println a + b |
||||||
|
} |
||||||
|
//调用方法的两种方式 |
||||||
|
show(2, 3) |
||||||
|
show 2, 3 |
||||||
|
``` |
||||||
|
|
||||||
|
可以看到,方法的调用是可以省略括号的,return 也是可以不写的,默认为最后一行代码的执行作为输出。 |
||||||
|
|
||||||
|
理解这一点是非常有用的,因为很多 build.gradle 脚本中的代码其实都是**方法调用**,比如 app module 的 build.gradle : |
||||||
|
|
||||||
|
```groovy |
||||||
|
android { |
||||||
|
compileSdkVersion 28 |
||||||
|
//... |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
它实际上就是调用的是: |
||||||
|
|
||||||
|
```groovy |
||||||
|
public void compileSdkVersion(int apiLevel) { |
||||||
|
compileSdkVersion("android-" + apiLevel); |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
#### Gradle 构建脚本基础 |
||||||
|
|
||||||
|
Gradle 构建脚本基础这一节,可以大致分为三块:Settings 和 Build 文件的作用,如何创建任务和配置任务依赖以及自定义属性的使用。 |
||||||
|
|
||||||
|
##### Settings 和 Build 文件 |
||||||
|
|
||||||
|
settings.gradle 用于初始化以及工程树的配置,放在根工程目录下。在 Gradle 中,多工程是通过工程树表示的,一个根工程是有很多子工程,一个子工程只有在 settings 文件里配置了 Gradle 才会识别,才会在构建的时候被包含进去: |
||||||
|
|
||||||
|
```groovy |
||||||
|
include ':app', ':crazy_plugin' |
||||||
|
``` |
||||||
|
|
||||||
|
当然,还可以使用其他方式: |
||||||
|
|
||||||
|
```groovy |
||||||
|
include ':example' |
||||||
|
project(':example').projectDir = new File(rootDir,'/chapter/example') |
||||||
|
``` |
||||||
|
|
||||||
|
我们可以为我们的子工程指定相应的目录,如果不指定,比如 app 和 crazy_plugin 子工程,默认目录就是其同级的目录。当然,我们也可以把工程放在任何目录下,可以非常灵活的对我们的工程进行分级、分类等,只要在 Settings 文件里面制定好路径就可以了。 |
||||||
|
|
||||||
|
每个 Gradle 工程都会有一个 build.gradle 文件,该文件是该 Project 构建的入口,可以在这里针对该 Project 进行配置,比如配置版本,需要哪些插件,依赖哪些库等。 |
||||||
|
|
||||||
|
既然每个 Project 都会有一个 build 文件,那么 root project 也不例外。root project 可以获取所有的 child project,所以可以在 root project 的 build 文件里对所有 child project 统一配置,比如应用的插件,依赖的 maven 中心库等。 |
||||||
|
|
||||||
|
```groovy |
||||||
|
//应用于所有的工程 |
||||||
|
allprojects { |
||||||
|
repositories { |
||||||
|
google() |
||||||
|
jcenter() |
||||||
|
} |
||||||
|
} |
||||||
|
//只应用于子工程 |
||||||
|
subprojects { |
||||||
|
apply plugin: 'java' |
||||||
|
repositories { |
||||||
|
google() |
||||||
|
jcenter() |
||||||
|
} |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
以上,不要以为 allprojects 和 subprojects 只能配置,它们只是两个方法,接受一个闭包作为参数,对工程进行遍历,遍历的过程中调用我们自定义的闭包,所以我们可以在闭包里配置、打印、输出或者修改 Project 的属性都可以。 |
||||||
|
|
||||||
|
##### 创建任务和任务依赖 |
||||||
|
|
||||||
|
定义一个 Task 是很简单的,以下给出了两种方式: |
||||||
|
|
||||||
|
```groovy |
||||||
|
task myTask { |
||||||
|
doLast { |
||||||
|
println 'myTask Start~' |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
tasks.create('myTaskV2'){ |
||||||
|
doLast{ |
||||||
|
println 'myTaskV2 Start~' |
||||||
|
} |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
第一种方式中,task 看起来像一个关键字,其实它是 Project 对象的一个函数,原型为 create(String name,Closure configureClosure),所以也就会有了第二种写法。 |
||||||
|
|
||||||
|
任务之间是可以有依赖关系的,这样我们就可以控制哪些任务先于哪些任务执行;哪些任务执行后,其他任务才能执行。比如我们运行 jar 任务之前,compile 任务一定要执行过,也就是说 jar 依赖于 compile; |
||||||
|
|
||||||
|
我们改一下上面的代码,让 myTaskV2 依赖 myTask: |
||||||
|
|
||||||
|
```groovy |
||||||
|
task myTask { |
||||||
|
doLast { |
||||||
|
println 'myTask Start~' |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
task myTaskV2(dependsOn: myTask) { |
||||||
|
doLast { |
||||||
|
println 'myTaskV2 Start~' |
||||||
|
} |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
这样,当执行 myTaskV2 的时候会先执行 myTask 任务。一个 Task 也是可以依赖多个 Task 的。 |
||||||
|
|
||||||
|
Project 在创建该任务的时候,同时把该任务对应的任务名注册为 Project 的一个熟悉,类型是 Task,这该如何理解呢,直接看代码: |
||||||
|
|
||||||
|
```java |
||||||
|
task myTask { |
||||||
|
doLast { |
||||||
|
println 'myTask Start~' |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
task myTaskV2(dependsOn: myTask) { |
||||||
|
doLast { |
||||||
|
println project.hasProperty('myTask') //true |
||||||
|
println project.hasProperty('myTask1') //false |
||||||
|
println 'myTaskV2 Start~' |
||||||
|
} |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
这也就说明了,每一个任务都是 Project 的一个属性。 |
||||||
|
|
||||||
|
最后想说一点,定义完 Task,是可以直接执行的,但我们也可以控制 Task 的开关,默认是可执行的,我们也可以把它关掉: |
||||||
|
|
||||||
|
```groovy |
||||||
|
myTaskV2.enabled = false |
||||||
|
``` |
||||||
|
|
||||||
|
##### 自定义属性 |
||||||
|
|
||||||
|
相对简单,就不多说了: |
||||||
|
|
||||||
|
```groovy |
||||||
|
//定义一个自定义属性 |
||||||
|
ext.nickname = 'Omooo' |
||||||
|
//定义多个自定义属性 |
||||||
|
ext { |
||||||
|
nicknameV2 = 'Omooo~' |
||||||
|
age = 18 |
||||||
|
} |
||||||
|
|
||||||
|
task myTaskV2() { |
||||||
|
doLast { |
||||||
|
println 'myTaskV2 Start~' |
||||||
|
println "${nickname}" |
||||||
|
println "${nickname}_${age}" |
||||||
|
} |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
#### Gradle 任务进阶 |
||||||
|
|
||||||
|
上面,我们已经掌握了如何去定义一个简单的 Task,这一节主要讲解 Task 的进阶用法,包括任务分组和描述、任务执行分析以及任务的 OnlyIf 断言。 |
||||||
|
|
||||||
|
##### 任务分组和描述 |
||||||
|
|
||||||
|
这是一个非常实用的技巧,还记得我们之前输出所有的 tasks 时是按照分组输出的嘛,我们自定义的 Task 也是可以定义分组和描述的,这便于我们查找。定义分组和描述非常简单: |
||||||
|
|
||||||
|
```groovy |
||||||
|
task myTaskV2 { |
||||||
|
group BasePlugin.BUILD_GROUP |
||||||
|
description '自定义的 myTaskV2' |
||||||
|
doLast { |
||||||
|
println 'myTaskV2 Start~' |
||||||
|
} |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
当我们执行 ./gradlew tasks 时就可以看到了: |
||||||
|
|
||||||
|
``` |
||||||
|
自定义的 Group tasks |
||||||
|
---------------- |
||||||
|
myTaskV2 - 自定义的 myTaskV2 |
||||||
|
``` |
||||||
|
|
||||||
|
##### 任务执行分析 |
||||||
|
|
||||||
|
我们知道,在 Task 中 doFirst 会首先执行,doLast 会最后执行,但有没有想过,内部是如何实现的呢? |
||||||
|
|
||||||
|
示例: |
||||||
|
|
||||||
|
```groovy |
||||||
|
Task task = task myTaskV2(type: CustomTask) |
||||||
|
task.doFirst { |
||||||
|
println 'CustomTask do first' |
||||||
|
} |
||||||
|
|
||||||
|
task.doLast { |
||||||
|
println 'CustomTask do last' |
||||||
|
} |
||||||
|
|
||||||
|
class CustomTask extends DefaultTask { |
||||||
|
|
||||||
|
@TaskAction |
||||||
|
def doSelf() { |
||||||
|
println 'CustomTask do self' |
||||||
|
} |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
其实原理很简单的,当我们执行 Task 的时候,其实就是执行其拥有的 actions 列表,这个列表保存在 Task 对象实例中的 actions 成员变量中,其类型是一个 List: |
||||||
|
|
||||||
|
```groovy |
||||||
|
//AbstractTask 类中 |
||||||
|
private List<ContextAwareTaskAction> actions = new ArrayList<ContextAwareTaskAction>(3) |
||||||
|
``` |
||||||
|
|
||||||
|
当我们以 TaskAction 注解标注的方法会作为 Task 的执行实体,实际上会调用以下方法: |
||||||
|
|
||||||
|
```groovy |
||||||
|
@Override |
||||||
|
public void prependParallelSafeAction(final Action<? super Task> action) { |
||||||
|
if (action == null) { |
||||||
|
throw new InvalidUserDataException("Action must not be null!"); |
||||||
|
} |
||||||
|
getTaskActions().add(0, wrap(action)); |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
接着就是 doFirst 和 doLast 了: |
||||||
|
|
||||||
|
```groovy |
||||||
|
@Override |
||||||
|
public Task doFirst(final Closure action) { |
||||||
|
hasCustomActions = true; |
||||||
|
if (action == null) { |
||||||
|
throw new InvalidUserDataException("Action must not be null!"); |
||||||
|
} |
||||||
|
taskMutator.mutate("Task.doFirst(Closure)", new Runnable() { |
||||||
|
public void run() { |
||||||
|
getTaskActions().add(0, convertClosureToAction(action, "doFirst {} action")); |
||||||
|
} |
||||||
|
}); |
||||||
|
return this; |
||||||
|
} |
||||||
|
|
||||||
|
@Override |
||||||
|
public Task doLast(final Closure action) { |
||||||
|
hasCustomActions = true; |
||||||
|
if (action == null) { |
||||||
|
throw new InvalidUserDataException("Action must not be null!"); |
||||||
|
} |
||||||
|
taskMutator.mutate("Task.doLast(Closure)", new Runnable() { |
||||||
|
public void run() { |
||||||
|
getTaskActions().add(convertClosureToAction(action, "doLast {} action")); |
||||||
|
} |
||||||
|
}); |
||||||
|
return this; |
||||||
|
} |
||||||
|
``` |
||||||
|
|
||||||
|
doFirst 也是添加到第一个,那为什么不是 doSelf 在 doFirst 之前执行呢?实际上,这是因为 Task 的创建的时候就已经执行 doSelf 了,然后创建 Task 成功执行才会执行 doFirst,所以最终会把 doFirst 的闭包放在 actions 的第一位。 |
||||||
|
|
||||||
|
##### 任务的 onlyIf 断言 |
||||||
|
|
Loading…
Reference in new issue