了解 Gradle 和 buildType Task,掌握这6大技能体系

void method(Closure closure){

}

method ({

println(“execute closure”)

})

上面就是Grovy的闭包定义语法,如果闭包为方法的最后一个参数,则可以省略圆括号,就像默认build.gradle文件内容一样。这样我们就知道原来这些不是配置,只是写法类似配置而已,其buildscript{}、repositories{} 等等都是方法调用,并且我们可以看看这些方法是怎么定义的。

/**

  • Configures the repositories for the script dependencies. Executes the given closure against the {@link

  • RepositoryHandler} for this handler. The {@link RepositoryHandler} is passed to the closure as the closure’s

  • delegate.

  • @param configureClosure the closure to use to configure the repositories.

*/

void repositorie

《Android学习笔记总结+最新移动架构视频+大厂安卓面试真题+项目实战源码讲义》

【docs.qq.com/doc/DSkNLaERkbnFoS0ZF】 完整内容开源分享

s(Closure configureClosure);

/**

  • Configures the repositories for the script dependencies. Executes the given closure against the {@link

  • RepositoryHandler} for this handler. The {@link RepositoryHandler} is passed to the closure as the closure’s

  • delegate.

  • @param configureClosure the closure to use to configure the repositories.

*/

void repositories(Closure configureClosure);

可以看到我上面说的语法一样,但是,其实,不过也确实有一些例外,比如:

classpath ‘com.android.tools.build:gradle:4.0.1’

这里的classPath跟其他的方法不一样,找不到这个方法,这其实是Grovy的一个机制,叫methodMissing,就是在方法调用的时候如果找不到方法,那么会去调用methodMissing()这个方法,最终在这个方法内处理逻辑。关于methodMissing这边不多做介绍,感兴趣的可以去了解下。

关于 buildTypes

buildTypes默认有两个方法debugrelease,但实际上是可以自定义的,命名随意,比如testing

buildTypes {

internal {

//TODO

}

debug {

//TODO

}

release {

minifyEnabled false

proguardFiles getDefaultProguardFile(‘proguard-android-optimize.txt’), ‘proguard-rules.pro’

}

}

}

从buildTypes的默认方法debug和release我们不难看出,这一般是用来区分生产和测试版本。 我们在编译时可以执行gradle assembleInternal/assembleDebug/assembleRelease命令,而执行这些命令不仅仅执行buildTypes下的这三个方法,如果在src目录下有internal/debug/release目录时,也会同时把对应目录打包到工程。比如执行 assembleInternal时会把src下的maininternal两个目录打包。我们可以用这个特性来对不同的app版本做差异化定制。

api,impementation,compileOnly

  • api 跟以前的compile完全一样,能传递依赖,即子module的依赖会带到主module

  • implementation 不传递依赖,只在当前module生效

  • compileOnly 只在编译时有效,不会参与打包

gradle wrapper

gradlew会根据gradle-wrapper.properties文件去下载对应的gradle版本

Task

我们来看一段gradle代码

println(rootProject.buildDir)

task clean(type: Delete) {

delete rootProject.buildDir

}

然后执行./gradlew

Configure project :

D:\work******\build

可以看到打印了build的目录

我们换一种写法,再试试

task clean(type: Delete) {

println(rootProject.buildDir)

delete rootProject.buildDir

}

然后再执行./gradlew 我没有执行clean task,所以正常是不应该打印的吧,我们来看看结果;

Configure project :

D:\work******\build

纳尼?什么鬼?没执行task也会执行task里面的代码么??但是为什么build目录没有被删除呢?这也太奇怪了吧!!

我相信多数人看到这里都会觉得很奇怪,方法难道能只执行某几行?其实并不是这样,clean task中的代码其实都执行了,只是在执行到delete的时候会把它放到task列表中,在调用clean之后再去真正执行删除逻辑。我们可以把添加到任务列表时的工作当作是配置操作,在调用clean时才是真正的执行操作

看到这有人就会想,那我如果要在task执行完毕之后输出提示日志,那该怎么做呢?如果在task最后执行,虽然日志输出了,但肯定不是任务执行之后输出的。那我们该怎么做呢?可以用doLast doFirst方法来打印执行时的日志

task clean(type: Delete) {

println(1)

delete rootProject.buildDir

println(2)

doLast {

println “执行完毕”

}

doFirst {

println “开始执行”

}

}

Configure project :

1

2

Task :clean

开始执行

执行完毕

从输出的日志也能看出task执行有两个阶段:配置执行

task调用其他task用denpendsOn

task taskA(){

doLast{

println"taskA executed"

}

}

task taskB(dependsOn:taskA){

doLast {

println “taskB executed”

}

}

这种写法表示执行taskB之前会执行taskA,如果taskA也有依赖,那也会先执行taskA的依赖,就这样一直递归。这就是一个依赖关系网,被称作有向无环图。

gradle执行流程

  • setting.gradle

  • 各个build.gradle的配置阶段 绘制有向无环图

  • 执行阶段

gradle plugin

一个简单的自定义plugin

class TestPlugin implements Plugin {

@Override

void apply(Project target) {

println “Hello gradle plugin”

}

}

apply plugin: TestPlugin

这样我们在执行./gradlew时,就能看到"Hello gradle plugin"了

Configure project :app

Hello gradle plugin

gradle plugin-extension

class TestPlugin implements Plugin {

上一篇:makefile基本语法


下一篇:大数据测试3