Android Gradle 入门到精通(一)

1.背景和意义

先后经历过多个Android开发团队,每个团队都有这样的一个人。他能在gradle中配置一些代码。导致最终的apk 根据品牌+渠道 命名。甚至还能根据不同品牌,配置不同的变量。根据不同变量展示不同界面。gradle除了能帮我们做上述的事之外,还能做哪些呢?

  • hook Android 编译过程
  • 配置和改善 Gradle 编译速度

学习完gradle,对android 项目的构建理解会更加深入;更加熟练的使用task,帮助我们构建项目

2.概念

gralde它就是一个构建工具,帮助我们编译,打包,发布,检查代码格式,下载jar包等等。它是基于groovy语言编写的。

问题1: android如何自动下载gradle
android 项目创建的时候,gradle文件中自带gradle-wrapper.jar 这个jar包会根据gradle-wrapper.properties的配置 下载对应的gradle 版本。

问题2: ./gradlewgradle的区别

gradlew就是对gradle的包装和配置,gradlew是gradle Wrapper,Wrapper的意思就是包装。 因为不是每个人的电脑中都安装了gradle,也不一定安装的版本是要编译项目需要的版本,那么gradlew里面就配置要需要的gradle版本

问题3:什么是Project, 什么是Task,什么是Extension
android studio 每个module都有一个build.gradle文件,对于这个build.gradle而言,project 就是我们的module。可以通过 gradle projects查看Project。

Extension 对我而言就是配置,初始化变量,这个变量的作用就是为Task提供参数。

Task 任务,也是Action,表示这个任务是行为。例如assembleRelease是一个Task,用来执行打包操作。打完的apk存放路径这个路径可以人为的配置。这个配置,在我看来就是拓展(Extension )

问题4:什么是Gradle Daemon
Gradle Daemon是gralde的守护进程。是一个长时间运行在后台的进程,避免频繁的初始化。并且 Gradle 会将 Project data 缓存在内存中,加快构建速度。使用Gradle Daemon 是没有任何副作用的,并且我们可以非常简单的使用它,不需要任何成本,Gradle Daomen 对我们用户是完全透明的。可以通过gradle --stop停止

问题4:什么是闭包Closure
Groovy中的闭包是一个开放,匿名的代码块,可以接受参数,返回值并分配给变量。如:

//定义一个代码块
def innerClosure = {
    printf("hello world")
}
//执行方式1
innerClosure()
//执行方式2
innerClosure.call()

闭包也能传递参数:

def innerClosure = {String s,Student student ->
    printf(s+student.toString())
}
innerClosure("hello world",new Student("chips","26"))

以上就是gralde的基本知识和概念

3.生命周期

gradle 的生命周期分为三个阶段:

  • 初始化阶段
  • 配置阶段
  • 执行阶段

3.1 初始化阶段

在初始化阶段,Gradle的主要任务是确定有哪些工程参与构建,会解析settings.gradle文件,为文件中参与构建的的项目都创建project对象。可以通过gradle init完成

3.2 配置阶段

在这个阶段,Gradle会分别在每个Project对象上执行对应的build.gradle脚本,对Project进行配置。gradle提供了俩个方法

void beforeEvaluate(Closure closure)  //Project 配置之前调用

void afterEvaluate(Closure closure)    //在 Project 配置结束后调用

其中需要注意的是:beforeEvaluate 必须在父模块的 build.gradle 对子模块进行配置才能生效,因为在当前模块的 build.gradle 中配置,它自己本身都没配置好,所以不会监听到。

3.3 执行阶段

在配置时,Gradle 决定了在执行阶段要运行的 task 的顺序,他们的依赖关系的内部结构被建模为一个有向无环图,我们可以称之为 taks 执行图,它可以用 TaskExecutionGraph 来表示。可以通过 gradle.taskGraph 来获取。

在 TaskExecutionGraph 中也可以设置一些 Task 生命周期的回调:

  • addTaskExecutionGraphListener(TaskExecutionGraphListener listener)
  • addTaskExecutionListener(TaskExecutionListener listener)
  • afterTask(Action action),afterTask(Closure closure)
  • beforeTask(Action action),beforeTask(Closure closure)
  • whenReady(Action action),whenReady(Closure closure)

4. 基础使用

4.1 Task的基础使用

task使用的官方文档:官方文档
build.gradle中task的定义:

task taskName(){
    //任务执行前
    doFirst {
        println "doFirst"
    }
    //任务执行完毕的打印
    doLast {
        println "doLast"
    }
}

dependsOn使用
task任务经常存在相互依赖关系。A task 依赖于 Btask 可以借助dependsOn。在hook的时候用的比较多

task B {
    println "this is a task B"
}
task A(dependsOn: B) {
    println "this is a task A"
}

type使用
type用来表示创建的task 的类型

class MyTask extends DefaultTask {
    public String message
    public void test(String s) {
        println("this is MyTask " + message)
    }
}

task testDe(type: MyTask) {
    message = "chips"
    println test()
}

task之间存在顺序性,使用到的api:mustRunAfter()shouldRunAfter()

task B {
    println "this is a task B"
}
task A(dependsOn: B) {
    println "this is a task A"
}

A.mustRunAfter(B)

这样运行A任务的话,也会顺带执行B任务


自定义插件注册Task的方式:

class MyPlugin implements Plugin<Project> {

    @Override
    void apply(Project project) {
        MyTask t = project.tasks.register("MyTask", MyTask).get()
        t.message="chips"
        println  t.toString()
    }
}

ps:千万别用project.getByName("MyTask")直接获取task,gradle编译出错

以上就是Task的基本用法,下面说说Extension的用法.

4.2 Extension 用法

什么是Extentsion?

就是 Gradle 的 Extension,翻译成中文意思就叫扩展。它的作用就是通过实现自定义的 Extension,可以在 Gradle 脚本中增加类似 android 这样命名空间的配置,Gradle 可以识别这种配置,并读取里面的配置内容。
一般我们通过ExtensionContainer来创建Extension,这个类跟TaskContainer命名有点类似。TaskContainer是用来创建并管理Task的,而ExtensionContainer则是用来创建并管理Extension的,通过Project的以下API可以获取到ExtensionContainer对象

在开始学习前,我们先看一个问题:
android app的build.gradle中 有这样的一段配置:

android {
    compileSdkVersion 31
    def versionCodeValue = 1
    def versionNameValue = "1.0"
}

这个android{} 就是一个名为android的Extension 。

我们来看看android这个Extension是如何定义的,系统源码连接:AppPlugin

AppPlugin插件:

  @Override
    void apply(Project project) {
        super.apply(project)
     ....
        extension = project.extensions.create('android', AppExtension,
                this, (ProjectInternal) project, instantiator,
                buildTypeContainer, productFlavorContainer, signingConfigContainer)
}

注册了一个名为android拓展,类型是AppExtension.

AppExtension:

public class AppExtension extends BaseExtension {
    final NamedDomainObjectContainer<DefaultProductFlavor> productFlavors
    final NamedDomainObjectContainer<DefaultBuildType> buildTypes
    final NamedDomainObjectContainer<SigningConfig> signingConfigs
    private final DefaultDomainObjectSet<ApplicationVariant> applicationVariantList =
        new DefaultDomainObjectSet<ApplicationVariant>(ApplicationVariant.class)
}

productFlavors/signingConfigs/buildTypes 分别是我们在build.gradle android下面配置的内容。他们都是一个又一个的拓展。用来配置参数,给Task的执行提供参数。

AppExtension 还有哪些拓展,可以访问AppExtension官网 看看还有哪些拓展可以写


使用Extension 分为三步:

  • 定义Extension 的参数类型
  • 在build.gradle中注册这个Extension
  • 实例化拓展,且给每个元素赋值
  1. 定义一个Extension 类型
class MyExtension{
    String name
    String age
}

2.注册Extension

getExtensions().create("MyExtension_拓展名称", MyInfoExtension)

3.在build.gradle 使用:

myExtension {
    name = "wxy"
    age = "123"
}

如何访问myExtension 呢?

project.afterEvaluate {
    MyInfoExtension mMyInfoExtension = getExtensions().getByName("myExtension")
    println "Extension 使用:  "+mMyInfoExtension.name+"   "+mMyInfoExtension.age
}

以上就是Extension 基本应用.

如何定义像android{}这样的Extension呢?

class Address{
    public String address
}
class MyExtension {
    public String name
    public String age
    public Address addressInfor=new Address();
    // 配置方法
    void addressConfig(Action<Address> action) {
        action.execute(addressInfor) // 直接执行 action 参数的 execute 方法,并传入扩展对象
    }
}

其中addressConfig 是必填项.
这样在build.gradle中就可以这样定义了 :

getExtensions().create("myExtension", MyExtension)
myExtension {
    name = "chips"
    age = "123"
    addressConfig{
        address="湖北"
    }
}

我们还可以在.groovy文件中,调用:

class MyPlugin implements Plugin<Project> {

    @Override
    void apply(Project project) {
        project.getExtensions().create("MyInfoExtension",MyInfoExtension)
    }
}

这一步做完之后,相当于我们注册了这样一个拓展。那么直接在buid.gradle中 进行声明就行.


4.3 属性的访问

属性的访问我遇到的有三种方式:

  • 根 build.gradle 中定义变量
  • 导入其他的.gradle,携带的变量
  • 访问gradle.properties
  • 访问local.properties
  1. 根build.gradle 定义的变量,其他的子build.gradle中都能访问。如:
project.ext {
    versionCode2=2021
}

project.ext 传入一个闭包,返回一个Defaultextrapropertiesextension 拓展,此类实际上是用一个Map保存变量的。

  1. 导入其他的.gradle
    类似定义constants.gradle
project.ext {
    min_sdk_version     = 24
    compile_sdk_version = 'android-31'
    target_sdk_version  = 31
    build_tools_version = "30.0.3"
    java_Compatibility  = JavaVersion.VERSION_1_8 // 支持lambda
}

然后在build.gradle中
apply from: "${rootProject.projectDir}/constants.gradle"

3.gradle.properties 访问
在gradle.properties中定义各种变量 如:

versionCode5=2022

在build.gradle中就可以这样使用:

defaultConfig {
     versionCode versionCode5  as int
}

必须得用as 转换成对应的类型,不然会报错

4.local.properties 访问
local元素的访问稍微麻烦点.得new 一个Properties 对象,指定加载的文件

Properties properties = new Properties()
if (project.rootProject.file('local.properties').exists()) {
    properties.load(project.rootProject.file('local.properties').newDataInputStream())
}

---
properties.getProperty("versionCode3") as int

之后通过properties.getProperty("versionCode3") as int 得到值。响应的也一定要加上as int,默认是string类型

以上就是gradle的基础知识。下一节讲Gradle的自定义插件

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

推荐阅读更多精彩内容