You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 

14 KiB

《Android Gradle 权威指南》读书笔记(一)基础知识

目录

  1. 思维导图
  2. 前言
  3. Gradle 基础知识
    • Gradle 命令行
    • Gradle 日志
    • Groovy 基础
  4. Gradle 构建脚本基础
    • Settings 和 Build 文件
    • 创建任务和任务依赖
    • 自定义属性
  5. Gradle 任务进阶
    • 任务分组和描述
    • 任务执行分析
    • 任务的 OnlyIf 断言
    • 执行其他 gradle 文件定义的方法

思维导图

前言

在写完 Gradle Plugin 入门指南Gradle Plugin 实践之 TinyPng Plugin 之后,总会有一些疑问,如果你仔细阅读过这两篇文章,你甚至会发现文章还有一些错误的说法,比如以下命令中 -q 参数的含义:

./gradlew -q build

-q 的含义并不是表示静默输出,而是…..(下面会讲到

《Android Gradle 权威指南》这本书理解起来相对比较简单,但是它解决了我的很多疑惑,我会把书中重要知识都在本文中记录下来。我希望读书是一种边解惑边吸收新知识的过程,我也正是以这种方式来写这篇文章的。

感谢作者的著作~,受益匪浅。

Gradle 基础知识

Gradle 基础知识分为三块:Gradle 命令行、Gradle 日志、Groovy 基础。

Gradle 命令行

Gradle 命令一般用于执行一些 Task 。

在前两篇文章,我们也接触到了两个 Gradle 命令,一个是用于执行我们自定义 task 的命令:

./gradlew myCustomTask

一个是用于执行上传 jar 包依赖的命令:

./gradlew uploadArchives

这个 uploadArchives Task 其实是 java plugin 里面的 Task,也就是说我们依赖了这个插件:

apply plugin: 'java'

起初我一直以为这个 task 是 android 里面的,在 app 模块中我们会依赖 android plugin:

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 androidDependencies

当然,我们也可以一次执行多个任务,比如:

./gradlew 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 是时候,验证是否执行成功是通过打印一句话来测试的:

println 'Gralde Task 执行完毕~'

使用 print 系列方法,把日志信息输出到标准的控制台输出流,它被 Gradle 定向为 QUIET 级别日志,但是 Gradle 是有日志 API 的:

logger.quiet("Quiet 信息")
logger.error("Error 信息")
logger.warn("Warn 信息")
logger.lifecycle("Lifecycle 信息")
logger.info('Info 信息')
logger.debug("Debug 信息")

这里其实是调用的 Project 的 getLogger() 方法获取的 Logger 对象的实例。

Groovy 基础

这一小节我并不会介绍过多的 Groovy 基础知识,因为完全没必要,大家看文档就好了。这里只说一点,那就是方法的调用。

//定义方法
def show(int a, int b) {
		println a + b
}
//调用方法的两种方式
show(2, 3)
show 2, 3

可以看到,方法的调用是可以省略括号的,return 也是可以不写的,默认为最后一行代码的执行作为输出。

理解这一点是非常有用的,因为很多 build.gradle 脚本中的代码其实都是方法调用,比如 app module 的 build.gradle :

android {
    compileSdkVersion 28
    //...
}

它实际上就是调用的是:

public void compileSdkVersion(int apiLevel) {
		compileSdkVersion("android-" + apiLevel);
}

Gradle 构建脚本基础

Gradle 构建脚本基础这一节,可以大致分为三块:Settings 和 Build 文件的作用,如何创建任务和配置任务依赖以及自定义属性的使用。

Settings 和 Build 文件

settings.gradle 用于初始化以及工程树的配置,放在根工程目录下。在 Gradle 中,多工程是通过工程树表示的,一个根工程是有很多子工程,一个子工程只有在 settings 文件里配置了 Gradle 才会识别,才会在构建的时候被包含进去:

include ':app', ':crazy_plugin'

当然,还可以使用其他方式:

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 中心库等。

//应用于所有的工程
allprojects {
    repositories {
        google()
        jcenter()
    }
}
//只应用于子工程
subprojects {
    apply plugin: 'java'
    repositories {
        google()
        jcenter()
    }
}

以上,不要以为 allprojects 和 subprojects 只能配置,它们只是两个方法,接受一个闭包作为参数,对工程进行遍历,遍历的过程中调用我们自定义的闭包,所以我们可以在闭包里配置、打印、输出或者修改 Project 的属性都可以。

创建任务和任务依赖

定义一个 Task 是很简单的,以下给出了两种方式:

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:

task myTask {
    doLast {
        println 'myTask Start~'
    }
}

task myTaskV2(dependsOn: myTask) {
    doLast {
        println 'myTaskV2 Start~'
    }
}

这样,当执行 myTaskV2 的时候会先执行 myTask 任务。一个 Task 也是可以依赖多个 Task 的。

Project 在创建该任务的时候,同时把该任务对应的任务名注册为 Project 的一个熟悉,类型是 Task,这该如何理解呢,直接看代码:

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 的开关,默认是可执行的,我们也可以把它关掉:

myTaskV2.enabled = false
自定义属性

相对简单,就不多说了:

//定义一个自定义属性
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 也是可以定义分组和描述的,这便于我们查找。定义分组和描述非常简单:

task myTaskV2 {
    group BasePlugin.BUILD_GROUP
    description '自定义的 myTaskV2'
    doLast {
        println 'myTaskV2 Start~'
    }
}

当我们执行 ./gradlew tasks 时就可以看到了:

自定义的 Group tasks
----------------
myTaskV2 - 自定义的 myTaskV2
任务执行分析

我们知道,在 Task 中 doFirst 会首先执行,doLast 会最后执行,但有没有想过,内部是如何实现的呢?

示例:

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:

//AbstractTask 类中
private List<ContextAwareTaskAction> actions = new ArrayList<ContextAwareTaskAction>(3)

当我们以 TaskAction 注解标注的方法会作为 Task 的执行实体,实际上会调用以下方法:

    @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 了:

    @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 断言

断言就是一个条件表达式,Task 有一个 onlyIf 方法,它接受一个闭包作为参数,如果改闭包返回 true 则该任务执行,否则跳过。这有很多用途,比如控制程序什么情况打什么样的包等。

比如:

task myCustomTask << {
    println 'myCustomTask Start~'
}

myCustomTask.onlyIf{
    project.hasProperty("build_type")
}

我们判断的条件是是否有 build_type 属性,所以当我们还是按往常执行如下命令:

./gradlew -q myCustomTask 

是不会有任何输出的,那我们怎么做才能会输出呢?

./gradlew -q -Pbuild_type=debug myCustomTask

加一个参数就好啦,就可以控制任务是否执行了。

命令行中 -P 的意思是为 Project 指定 K-V 格式的属性键值对,使用格式为 -PK=V。

执行其他 gradle 文件里面的方法

有时候,我们想要在当前 gradle 文件里面,去执行其他 gradle 里面定义的方法。

比如我们想要在 app.gradle 定义一个 Task,去执行放在根目录下的 dep.gradle 文件里面定义的方法,做法如下:

dep.gradle:

gradle.ext {
    showDep = this.&showDep
}
def showDep() {
    println("showDep")
}

app.gradle:

apply from: "../dep.gradle"
task("appTask") {
    doLast {
        println("appTask")
        gradle.ext.showDep()
    }
}