美文网首页gradle
Android Gradle(三)- gradle

Android Gradle(三)- gradle

作者: Stan_Z | 来源:发表于2020-04-10 00:53 被阅读0次

上一篇对groovy语法与核心api做了简单总结,为gradle具体业务功能实现做了语言铺垫,那么接下来进入到gradle api的学习。

一、gradle生命周期

gradle生命周期分为三个阶段:
Initialization初始化阶段: 各工程初始化,解析settings.gradle,构建所有project对应的project对象。
Configuration配置阶段: 执行所有工程中build.gradle的配置代码,解析所有project对象中的task,构建task的拓扑图。Task的执行顺序在配置阶段就确定好了。
Execution执行阶段: 执行task及其依赖task。

gradle部分生命周期监听回调

//配置阶段开始前的监听回调
this.beforeEvaluate {
}

//配置阶段完成以后的回调
this.afterEvaluate {
}

//当前project build完成
this.gradle.buildFinished {
}

//project构建之前
this.gradle.beforeProject {
}

//project构建之后
this.gradle.afterProject {
}

//其他监听
this.gradle.addBuildListener(new BuildListener() {
    @Override
   void buildStarted(Gradle gradle) {
    }

    @Override
   void settingsEvaluated(Settings settings) {
    }

    @Override
   void projectsLoaded(Gradle gradle) {
    }

    @Override
   void projectsEvaluated(Gradle gradle) {
    }

    @Override
   void buildFinished(BuildResult result) {
    }
})
二、 project

gradle-5.4.1/src/core-api/org/gradle/api/Project.java
gradle中根工程和各module对应一个project,它是脚本入口。

project相关api
this.getProject() or this.project//获取project
this.getRootProject() or this.rootProject//获取根project
this.getSubprojects() or this.subprojects//获取所有子project 根project获取的子project即各个module
this.getParent() or this.parent//获取父project

在build.gradle中方法的定义及调用
this.helloGradle()

def helloGradle() {
    def allprojects = this.getAllprojects()
    allprojects.each { project ->
        println 'project name :' + project.name
   }
}

//对app project 进行配置
project('app') { Project project ->
   apply plugin: ‘...'
   group ‘...'
   version ‘...'
   dependencies {...}
   android {…}
   ...
}
属性相关api

Project自带属性

public interface Project extends Comparable<Project>, ExtensionAware, PluginAware {
    /**
    * The default project build file name.
    */
   String DEFAULT_BUILD_FILE = "build.gradle";
  
 /**
    * The hierarchy separator for project and task path names.
    */
   String PATH_SEPARATOR = ":";

   /**
    * The default build directory name.
    */
   String DEFAULT_BUILD_DIR_NAME = "build";
   String GRADLE_PROPERTIES = "gradle.properties";
   String SYSTEM_PROP_PREFIX = "systemProp";
   String DEFAULT_VERSION = "unspecified";
   String DEFAULT_STATUS = "release";
}

//使用ext闭包拓展属性
ext {
    compileSdkVersion = 29
}
//这里修改compileSdkVersion为例
Android{
    compileSdkVersion [this.project.compileSdkVersion](http://this.project.compilesdkversion/)
}

//在gradle中引入文件
apply from :this.file('common.gradle’)

//在gradle中引入插件
apply plugin:'gradle-plugin’

//property基本操作
getProperty(‘')
setProperty('')
hasProperty(‘')

另外还可以在gradle.properties中配置 key - value形式的全局属性
#key-value
isLoadTest=true
# 这里可以以key-value的形式定义全局的属性,使用的时候需要用toBoolean() toInteger()等转换
# 另外属性命名不能与已有方法重名,这样容易找不到属性

//三方库引入方式对比
2.X
compile: 编译运行期都参与。
provided: 编译期参与,运行时不参与
3.0之后:
compile 延伸出api 和 implementation。
api : 与compile效果一样。
implementation : implementation依赖的库只能自己库本身访问,例如:A依赖B,B依赖C,如果B依赖C是使用的implementation依赖,那么在A中是访问不到C 中的方法的,如果需要访问,请使用api依赖。其他与compile一致。
compileOnly: 与provided效果一样。
其他:
runtimeOnly: 和apk效果一样,打包时有效,不参与编译。
testImplementation:和testCompile效果一样,在单元测试和打包测试apk的时候有效。
debugImplementation:和debugCompile效果相同, 在debug模式下有效。
releaseImplementation:和releaseCompile效果相同,只在release模式和打包release包情况下有效。
文件相关api

路径:

//工程根目录
getRootDir().absolutePath //or this.rootDir

//工程目录/build
getBuildDir().absolutePath //or this.buildDir

//module目录
getProjectDir().absolutePath //or this.projectDir

//获取文件
getFile('build.gradle')
def getFile(String path) {
    try {
        //new file要传入绝对路径,file()传入相对路径,相对于当前project开始查找
       def file = file(path)//project的file方法获取文件,files方法定位多个符合条件文件
       return file.text
   } catch (GradleException e) {
        println 'file not found'
   }
    return null
}

//文件拷贝 from into
copy {
    from file('build/outputs/apk/')
    into getRootProject().getBuildDir().path + '/apk/'
   //附加功能
   exclude {} //包含
   rename {} //重命名
}

//对文件树进行遍历
fileTree('build/outputs/apk/') { FileTree filetree ->
    filetree.visit { FileTreeElement element ->
        println 'the file name is:' + element.file.name
       copy {
            from element.file
           into getRootProject().getBuildDir().path + 'test'
       }
    }
}
其他
buildscript { ScriptHandler scriptHandler ->
    //gradle本身配置工程的仓库地址
   scriptHandler.repositories {}
    //gradle本身配置工程的三方库依赖地址
   scriptHandler.dependencies {}
    //对应project中的dependencies,是配置应用程序三方库的依赖
}
三、task

gradle-5.4.1/src/core-api/org/gradle/api/Task.java
一个Task表示构建的单个原子动作。构建工作就是由各种Task构成的。

新建task两种写法:
//用task函数创建
task myTask {
    println 'hello task'
}

//使用taskContainer容器创建
this.tasks.create(name: 'myTask') {
    println 'hello task'
}

task所有可配置信息
public interface Task extends Comparable<Task>, ExtensionAware {
   String TASK_NAME = "name";
   String TASK_DESCRIPTION = "description";
   String TASK_GROUP = "group";
   String TASK_TYPE = "type";
   String TASK_DEPENDS_ON = "dependsOn";
   String TASK_OVERWRITE = "overwrite";
   String TASK_ACTION = "action";
...
}

task在执行阶段的两个回调:
doFirst{}//在已有的task之前去添加逻辑
doLast{}//在已有的task之后去添加逻辑

调用方式:
task myTask {
    doLast{ //内部调用
    }
}
//or
myTask.doLast{ //外部调用
}

小实例:计算prebuild到build的总时长
def startTime, endTime
//配置阶段执行完以后
this.afterEvaluate { Project project ->
    //保证要找的task配置完成
   def preBuildTask = project.tasks.getByName('preBuild')
    preBuildTask.doFirst {
        startTime = System.currentTimeMillis()
    }

    def buildTask = project.tasks.getByName('build')
    buildTask.doLast {
        endTime = System.currentTimeMillis()
        println "the build time is  ${endTime - startTime}"
   }
}

gradle在配置阶段会构建task的拓扑图,这个过程组织好task的执行顺序
task依赖

dependsOn配置依赖

task taskX {
   doLast {
       println 'taskX'
   }
}

静态配置
//taskY在taskX之后执行
task taskY(dependsOn:taskX) {
   doLast {
       println 'taskY'
   }
}

//or
task taskY {
    dependsOn taskX
    doLast {
        println 'taskY'
   }
}

//or
taskY.dependsOn taskX

指定依赖之后,执行taskY,会先执行它依赖的taskX ,然后执行它自己。

动态配置
task taskZ() {
    dependsOn this.tasks.findAll { task ->
        return task.name.startsWith('lib')//动态依赖lib开头的task
   }
    doLast {
        println 'taskZ'
   }
}
task执行顺序

mustRunAfter()、shouldRunAfter()
只指定顺序执行,没有依赖关系。

task taskZ {
   mustRunAfter taskX //强制task在指定的task之后执行
   shouldRunAfter taskX //不强制性的
   doLast {
        println 'taskZ'
   }
}

如果单独执行taskZ,那么就只会执行taskZ,跟taskX和taskY无关,只有taskX、taskY、taskZ都执行的时候,这个条件能约束执行顺序。

finalizedBy

taskX.finalizedBy taskY //执行完taskX之后会拉起执行taskY

总结:dependsOn指定依赖关系,先执行依赖的task再执行自己;finalizedBy是当前task执行完成拉起让自己结束的task;而mustRunAfter是多个执行的task之前约束一个执行顺序,默认情况下无依赖关系的task执行顺序是随机的。

task输入输出

TaskInputs、TaskOutputs

输入输出在Gradle中用于增量构建,执行一个任务,如果在编译时,gradle判断在从上一次编译中,该task的输入输出没有任何改变,那么gradle就会跳过该task,前提是这个task至少有一个输入或输出。

支持的输入输出类型参考TaskInputs接口
inputs.file/files/property
outputs.file/files

四、Settings

在gradle中,通过执行Settings类来完成构建的初始化阶段。
最重要的是inlcude方法

include ':app', ':lib_network’ //配置工程子project
rootProject.name = ‘GradleDemo'

这里可以做条件判断是否引入子project
if (hasProperty('isLoadTest') ? isLoadTest.toBoolean() : false) {
    include ':Test'
}

对应两个监听
gradle.settingsEvaluated {
   println “Settings.gradle 初始化执行结束"
}

gradle.projectsLoaded {Gradle gradle ->
   println “Settings.gradle所有在 settings 中 include 的 Project 对象都创建完成了"
}
五、SourceSet
//5.4.1版本配置方法
android {
    sourceSets {
        // sourceSets的main对象,该对象属于默认对象。
        main {
            resources {
                srcDirs 'src/main/res', 'src/main/res-ad', 'src/main/res-player'
           }
        }
    }
}

//3.0版本配置方法
// 配置源码路径。这个sourceSets是java插件引入的
sourceSets {
   // 还可以根据productFlavors中的其他渠道配置对应的sourceSets对象。编译时,会首先加载对应productFlavors中的资源,没有则加载main中默认资源。
   main {
        manifest.srcFile 'AndroidManifest.xml'// 这是一个函数
       java.srcDirs = ['src']
        resources.srcDirs = ['src']
        aidl.srcDirs = ['src']
        renderscript.srcDirs = ['src']
        res.srcDirs = ['res']
        assets.srcDirs = ['assets']
        jniLibs.srcDirs = ['libs']
    }

groovy和gradle文章写下来,基本是代码的形式,可能阅读并不是那么直观,但是这个比较符合我的习惯。方便查询吧。

相关文章

网友评论

    本文标题:Android Gradle(三)- gradle

    本文链接:https://www.haomeiwen.com/subject/kinkmhtx.html