Android构建04-Gradle构建流程

构建的生命周期

Gradle项目的构建分为三个阶段:初始化、配置、执行。
参考官方手册 Build Lifecycle

初始化(Initialization)

在这个阶段中,Gradle决定哪些项目加入到构建中(因为Gradle支持多项目构建),并为这些项目分别创建一个Project实例。

Gradle支持单项目和多项目构建,如果是单项目构建,则初始化当前项目即可,如果是多项目构建,则需要决定哪些项目需要加入到构建中并初始化。那么:

  1. Gradle是如何判断当前构建是单项目还是多项目构建呢?
  2. 如果是多项目构建,Gradle又是如何决定将哪些项目加入到构建中呢?

多项目还是单项目构建?
这里的关键是一个名为setting.gradle的文件,Gradle在初始化阶段会首先去查找setting.gradle文件,查找的规则如下:

  1. 查找当前构建目录下的setting.gradle文件。
  2. 如果没有找到,则去与当前目录有相同嵌套级别的master目录查找。
  3. 如果没有找到,则去父目录查找。
  4. 如果没有找到,则进行单项目构建。
  5. 如果找到了,Gradle去检查当前项目在settings.gradle中是否有定义。如果没有,则进行单项目构建,否则进行多项目构建。

多项目工程在根目录必须存在setting.gradle文件,单项目工程则可以不需要这个文件。

多项目构建的项目集如何决定?
多项目构建时,项目集由setting.gradle创建。项目集可以用一个树形结构表示,每个节点表示一个项目,并且有对应的项目路径。在大多数情况下,项目路径与文件系统中项目的位置一致,当然这也是可以配置的。
setting.gradle中,有很多方法可以定义项目树,常见的是层次化和扁平化的布局。

层次化布局

// setting.gradle
include 'project1', 'project2:child', 'project3:child1'

include方法接收多个项目路径作为参数。
每个项目路径对应文件系统中的一个目录,通常,项目路径和文件目录一一对应,比如project2:child对应项目根目录下的project2/child目录。
如果项目树的叶子节点和它的父节点都需要包含在构建中,则只需指定叶子节点即可。比如,project2:child,将创建两个项目实例project2project2:child

扁平化布局

// setting.gradle
includeFlat 'project3', 'project4'

includeFlat方法将目录名作为参数。这些目录是根项目目录的兄弟目录,并作为多项目树中根项目的子项目。

Settings语法
这里主要说一下include方法的语法,其余的语法请见对应的DSL Referencesetting.gradle在构建时会创建代理对象Settings,其include方法对应着 Settings.include(java.lang.String[]),查看其DSL Reference:

  1. include方法接收项目路径数组作为参数,并将对应的项目添加到构建中。
  2. 参数中支持的项目路径分隔符为“:”,而不是“/”。
  3. 路径的最后一个节点是项目的名称。
  4. 项目路径是根项目目录下的相对路径,可以使用ProjectDescriptor.setProjectDir(java.io.File))更改。

示例

// 引入两个项目, 'foo' and 'foo:bar'
// 对应的文件目录是 $rootDir/a 和 $rootDir/a/b
// directories are inferred by replacing ':' with '/'
include 'foo:bar'

// include one project whose project dir does not match the logical project path
// 引入‘baz’项目,对应的文件目录是foo/baz
include 'baz'
project(':baz').projectDir = file('foo/baz')

配置(Configuration)

在这个阶段,Gradle会配置加入到构建中的Project实例,执行所有项目对应的构建脚本。

作为名词的‘配置(Configuration)'是指,Gradle执行项目的构建脚本这一过程。
作为动词的'配置( Configure)'是指,Gradle执行项目的构建脚本这一动作。

默认配置模式

默认情况下,Gradle会配置settings.gradle里的所有项目,不论这些项目与最终执行的任务是否有关系。这样做是因为Gradle允许一个项目在配置和执行阶段访问任何其他项目,如Gradle里可以进行跨项目配置(Cross project configuration),一个项目的配置可能会依赖其他项目,所以在执行任务之前,需要配置所有的项目。

项目配置按照广度( breadth-wise)顺序来执行,如父项目先于子项目被配置。

按需配置模式

由于多项目配置中,可能存在大量无需配置的项目,如果需要配置所有项目后才执行任务则会浪费大量的时间。从Gradle1.4开始,有一个孵化中的特性,叫做按需配置(configuration on demand)模式。按需配置项目时,Gradle只配置与最终任务相关联的项目,以缩短构建时间。这个模式以后可能会成为默认模式,甚至成为唯一的模式。截至我写这篇文章(Gradle发布4.10)时,按需配置模式还只是一个实验中的特性,不能保证所有的构建都能正确运行,尤其是存在耦合的多项目构建。

项目的耦合和解耦请见文档的Decoupled Projects。一个简单的耦合项目就是在子项目中使用allprojects and subprojects等方法,在根项目中使用这两个方法则不会影响按需配置模式。

按需配置模式下,项目配置遵循规则如下:

  • 根项目总会被配置。
  • 构建的当前项目也会被配置。
  • 项目的依赖会被配置。如果项目A将项目B作为依赖,则构建A时,A和B都会被配置。
  • 任务依赖对应的项目会被配置。如’someTask.dependsOn(":someOtherProject:someOtherTask")‘,someOtherProject会被配置。
  • 通过命令行构建任务时会配置相应的项目。如构建 'projectA:projectB:someTask'时会配置projectB。

进行构建时,可以在命令行加入--configure-on-demand来指定用按需配置模式来进行构建,如gradle hello --configure-on-demand将以按需配置模式执行‘’hello“任务。

执行(Execution)

在这个阶段,首先,Gradle确定要执行的任务集,任务集是由输入到命令行的任务名称参数和当前目录决定的。然后,Gradle会去执行任务集中的每个任务。
任务(Task)是由一系列的活动(Action)组成的,当任务执行的时候,活动会依次执行。可以通过doFirstdoLast方法将活动添加到任务中。

问题:

  1. 多项目构建中,Gradle如何确定任务是哪个项目的?尤其是任务名称相同的情况下。
  2. 如果确定多个任务执行的顺序?

任务的位置
$ gradle hello为例,在执行阶段,Gradle会从构建的当前项目目录开始,依据项目树往下查找名称为hello的任务并执行。因此Gradle不会执行当前项目的父项目和兄弟节点项目的任务。

任务的顺序
如果没有额外配置,Gradle将以字母数字顺序执行任务。比如 “:consumer:action” 将先于 “:producer:action”执行。

If nothing else is defined, Gradle executes the task in alphanumeric order


示例-单项目构建

# 创建项目目录
$ mkdir single-project-build-lifecycle-demo
$ cd single-project-build-lifecycle-demo

# 创建settings.gradle文件
$ vi settings.gradle

settings.gradle文件

rootProject.name = 'single-project-build-lifecycle-demo'
println 'settings.gradle -> this is executed during the initialization phase.'
# 创建build.gradle文件
$ vi build.gradle

build.gradle文件

println 'build.gradle -> global println -> This is executed during the configuration phase.'

task configured {
    println 'build.gradle -> task configured -> This is also executed during the configuration phase.'
}

task test {
    doLast {
        println 'build.gradle -> task test -> doLast -> This is executed during the execution phase.'
    }
}

task testBoth {
    doFirst {
      println 'build.gradle -> task testBoth -> doFirst -> This is executed first during the execution phase.'
    }
    doLast {
      println 'build.gradle -> task testBoth -> doLast -> This is executed last during the execution phase.'
    }
    println 'build.gradle -> task testBoth -> This is executed during the configuration phase as well.'
}

开始构建,并执行testBoth任务 $ gradle testBoth,输出如下:

settings.gradle -> this is executed during the initialization phase.

> Configure project :
build.gradle -> global println -> This is executed during the configuration phase.
build.gradle -> task configured -> This is also executed during the configuration phase.
build.gradle -> task testBoth -> This is executed during the configuration phase as well.

> Task :testBoth
build.gradle -> task testBoth -> doFirst -> This is executed first during the execution phase.
build.gradle -> task testBoth -> doLast -> This is executed last during the execution phase.

BUILD SUCCESSFUL in 0s
1 actionable task: 1 executed

构建过程如下:

  1. 初始化阶段。
    执行settings.gradle脚本,所以首先输出'settings.gradle -> this is executed during the initialization phase.'。
  2. 配置阶段。
    由于settings.gradle没有定义项目,根项目即当前项目有build.gradle,因此只是执行这个文件。顺序执行:打印语句、三个task方法。
    这里的三个task方法对应Project实例中Task task(String name, Closure configureClosure),查看其Reference文档,说是这个方法会产生一个任务对象并添加到当前Project实例中,并且在返回这个任务的时候会执行闭包以配置这个任务。也就说在配置我们这个项目时,三个task方法的闭包都会被执行。
    以testBoth任务对应的task方法为例,它的闭包里包含doFirstdoLastprintln三个方法。其中doFirstdoLastTask的方法,查看doFirstReference文档,说是添加闭包到任务的action列表中,当任务执行的时候,会执行这个闭包。在配置阶段,并不会去执行任务,也就是说不会去执行doFirstdoLast闭包里的内容。
    因此,在配置阶段,Gradle会依次执行三个任务方法里的闭包,而不会执行闭包里添加action的闭包即doFirstdoLast的闭包,因此会依次输出
    'build.gradle -> task configured -> ...' 和‘build.gradle -> task testBoth -...’
  3. 执行阶段。
    由于在命令行只构建testBoth任务,因此这个阶段只执行testBoth任务中的Action列表,依次执行doFirstdoLast的闭包。


示例-常见的多项目构建

参考官方手册Authoring Multi-Project Builds
模拟一个常见的Java项目,目录结构如下

multi-project-build-lifecycle-demo/
  settings.gradle
  build.gradle
  api/
    src/main/java/
      org/gradle/sample/
        api/
          Person.java
        apiImpl/
          PersonImpl.java
  services/personService/
    src/
      main/java/
        org/gradle/sample/services/
          PersonService.java
      test/java/
        org/gradle/sample/services/
          PersonServiceTest.java
  shared/
    src/main/java/
      org/gradle/sample/shared/
        Helper.java

根目录settings.gradle

include 'api', 'shared', 'services:personService'

根目录build.gradle

subprojects {
    apply plugin: 'java'
    group = 'org.gradle.sample'
    version = '1.0'
    repositories {
        mavenCentral()
    }
    dependencies {
        testCompile "junit:junit:4.12"
    }
}

project(':shared') {
    task hello {
        doLast {
            println "shared task hello -> doLast"
        }
        println "shared task hello"
    }
}

project(':api') {
    dependencies {
        compile project(':shared')
    }
    task hello {
        doLast {
            println "api task hello -> doLast"
        }
        println "api task hello"
    }
}

project(':services:personService') {
    dependencies {
        compile project(':shared'), project(':api')
    }
    task hello {
        doLast {
            println "personService task hello -> doLast"
        }
        println "personService task hello"
    }
}

执行命令gradle -i :services:personService:hello,输出如下

-i 是gradle的log等级的一种,输出较为详细的信息

Initialized native services in: /Users/zhangliang/.gradle/native
The client will now receive all logging from the daemon (pid: 27188). The daemon log file: /Users/zhangliang/.gradle/daemon/4.8.1/daemon-27188.out.log
Starting 4th build in daemon [uptime: 2 mins 55.761 secs, performance: 98%, no major garbage collections]
Using 8 worker leases.
Starting Build
Settings evaluated using settings file '/Users/zhangliang/Projects/tools/learn-gradle/multi-project-build-lifecycle-demo/settings.gradle'.
Projects loaded. Root project using build file '/Users/zhangliang/Projects/tools/learn-gradle/multi-project-build-lifecycle-demo/build.gradle'.
Included projects: [root project 'multi-project-build-lifecycle-demo', project ':api', project ':services', project ':shared', project ':services:personService']

> Configure project :
Evaluating root project 'multi-project-build-lifecycle-demo' using build file '/Users/zhangliang/Projects/tools/learn-gradle/multi-project-build-lifecycle-demo/build.gradle'.
shared task hello
api task hello
personService task hello

> Configure project :api
Evaluating project ':api' using build file '/Users/zhangliang/Projects/tools/learn-gradle/multi-project-build-lifecycle-demo/api/build.gradle'.

> Configure project :services
Evaluating project ':services' using build file '/Users/zhangliang/Projects/tools/learn-gradle/multi-project-build-lifecycle-demo/services/build.gradle'.

> Configure project :shared
Evaluating project ':shared' using build file '/Users/zhangliang/Projects/tools/learn-gradle/multi-project-build-lifecycle-demo/shared/build.gradle'.

> Configure project :services:personService
Evaluating project ':services:personService' using build file '/Users/zhangliang/Projects/tools/learn-gradle/multi-project-build-lifecycle-demo/services/personService/build.gradle'.
All projects evaluated.
Selected primary task ':services:personService:hello' from project :services:personService
Tasks to be executed: [task ':services:personService:hello']
:services:personService:hello (Thread[Task worker for ':',5,main]) started.

> Task :services:personService:hello
Task ':services:personService:hello' is not up-to-date because:
  Task has not declared any outputs despite executing actions.
personService task hello -> doLast
:services:personService:hello (Thread[Task worker for ':',5,main]) completed. Took 0.001 secs.

BUILD SUCCESSFUL in 0s
1 actionable task: 1 executed

从输出可以看出:

  1. Gradle首先查找settings.gradle文件进行初始化,确定根项目目录并创建所有include的项目实例。其中包括services
  2. 配置settings.gradleinclude的所有的项目。项目配置的顺序,首先是根项目,其次是子项目,多个子项目看来也是按照字母顺序依次进行的。

当前配置排序是
[root project 'multi-project-build-lifecycle-demo', project ':api', project ':services', project ':shared', project ':services:personService']
当我将‘shared’项目重命名为‘ashared’后,配置排序变为
[root project 'multi-project-build-lifecycle-demo', project ':api', project ':ashared', project ':services', project ':services:personService']

  1. 最后执行任务:services:personService:hello


示例-按需配置的多项目构建

参考文档Build script of water (parent) project

创建一个多项目工程,目录结构如下:

.
├── bluewhale/
├── build.gradle
├── krill/
└── settings.gradle

settings.gradle文件

rootProject.name = 'water'
include 'bluewhale', 'krill'

build.gradle文件

allprojects {
    task hello {
        doLast { task ->
            println "I'm $task.project.name"
        }
    }
}

根目录下执行gradle -i :bluewhale:hello,输出如下(省略部分无关信息):

Initialized native services in: /Users/zhangliang/.gradle/native
Included projects: [root project 'water', project ':bluewhale', project ':krill']

> Configure project :
> Configure project :bluewhale
> Configure project :krill
All projects evaluated.

> Task :bluewhale:hello
I'm bluewhale

从输出可以看出,所有的项目都进行了配置。

根目录下执行gradle -i :bluewhale:hello --configure-on-demand,输出如下(省略部分无关信息):

Initialized native services in: /Users/zhangliang/.gradle/native
Included projects: [root project 'water', project ':bluewhale', project ':krill']
Configuration on demand is an incubating feature.

> Configure project :
> Configure project :bluewhale
All projects evaluated.

> Task :bluewhale:hello
I'm bluewhale

上面的输出中出现了一句话"Configuration on demand is an incubating feature",表示当前的“按需配置模式”还只是实验中的功能。从输出中可以看出,没有配置与任务无关的“krill”项目。


参考

  1. Gradle官网


上篇:学习Android构建系统(三)-Gradle基础
下篇:学习Android构建系统(五)-GradleWrapper、命令行和环境配置

提示:这篇文章也可以见我的个人博客

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

推荐阅读更多精彩内容

  • 说明 本文主要从实现原理和代码层面介绍Gradle开发相关知识。关于本文中提到的、Gradle中的基本概念等内容,...
    jzj1993阅读 7,915评论 1 33
  • 说明 本文主要介绍和Gradle关系密切、相对不容易理解的配置,偏重概念介绍。部分内容是Android特有的(例如...
    jzj1993阅读 15,623评论 1 62
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,053评论 25 707
  • 类的定义 [修饰符] class 类名{ } 示例代码 定义类注意事项: 如果类名使用了public修饰符,必须保...
    若愚同学阅读 105评论 0 2
  • 有人见过你最真实的脸 劝你戴上了面具 后来有人细瞧过你的样子 说想看看你的脸 你龇牙咧嘴的摘下了一层 所有的人都跑...
    橙se火机阅读 175评论 0 1