关于AS中的Gradle

2018-02-08  本文已影响714人  34sir

Gradle是干嘛的?

简单的讲,Gradle是基于Groovy的项目自动化构建工具,用来声明项目配置
具体一点,在开发过程中主要实现工程的管理,例如依赖,打包,部署,发布,各种渠道的差异管理,Gradle负责将这些行为以代码的方式描述出来以实现行为的复用

基本概念

projects , tasks and action:工程,任务,还有行为
一个项目至少要有一个工程,一个工程至少要有一个任务,一个任务由一些行为组成
结合实例来理解:
对于一个标准的AS项目而言,一个build.gradle对应一个project,action则相当于build.gradle中的一段段代码块
在工程构建的过程中,gradle会根据build.gradle中的配置信息生成相应的project和task
Project实质上是一系列task的集合,每一个task执行一些工作,比如编译类文件,解压缩文件,删除文件等等

怎么用?

构建

task

task标示一个逻辑上的执行单元
举几个栗子:

自定义task

在build.gradle中定义

task haha {  
    println "HAHA"  
}  

执行gradle haha,则会打印出haha

task hello << {  
    println "hello,world"  
}  

<<意思是给hello这个task添加一些action,其实就是调用了task的doLast方法,相当于以下:

task hello {  
    doLast{  
        println "hello,world"  
    }  
}  

目前build.gradle中存在haha和hello两个task,我们执行一下gradle看看结果:

ckc@ckc-PC:~/AopDemo$ ./gradlew
Picked up _JAVA_OPTIONS:   -Dawt.useSystemAAFontSettings=gasp

> Configure project :
HAHA
The Task.leftShift(Closure) method has been deprecated and is scheduled to be removed in Gradle 5.0. Please use Task.doLast(Action) instead.
        at build_1wojgokbha0tm9swmixbdcemv.run(/home/ckc/AopDemo/build.gradle:33)

结果可知,只执行了haha的task
这里就有两个疑问:
1.并没有执行haha的task,他为什么执行了?
2.既然haha执行了hello为神马没有执行?
解释是,方式1的定义方式生成的task是在初始化阶段就会默认执行,而方式2是在执行阶段才会执行
简单的讲就是doLast起到的作用

// 需要继承自DefaultTask
class HelloTask extends DefaultTask {
    // @Optional 表示在配置该Task时,message是可选的。
    @Optional
    String message = 'I am 34sir'
    // @TaskAction 表示该Task要执行的动作,即在调用该Task时,hello()方法将被执行
    @TaskAction
    def hello() {
        println "$message"
    }

    @Override
    void onlyIf(Spec<? super Task> onlyIfSpec) {

    }

    @Override
    void setOnlyIf(Spec<? super Task> onlyIfSpec) {

    }

    @Override
    Task doFirst(Action<? super Task> action) {
        return null
    }

    @Override
    Task doLast(Action<? super Task> action) {
        return null
    }

    @Override
    int compareTo(Task task) {
        return 0
    }

    @Override
    void setActions(List<Action<? super Task>> actions) {

    }
}

// hello使用了默认的message值
task helloOne(type: HelloTask)

// 重新设置了message的值
task helloTwo(type: HelloTask) {
    message = "你站在这别动,我去给你买橘子"
}

当前工程中定义

我们可以把task定义在buildSrc module中:
在项目的根目录下新建一个名为buildSrc文件夹(一定要这样命名),然后依次新建子目录src/main/groovy,然后可以建自己的包名,这里以com.oil为例,依次新建子目录demo/gradle/task,然后在buildSrc根目录下新建build.gradle文件,此build.gradle中:

 apply plugin: 'groovy'

 dependencies {
    compile gradleApi()
    compile localGroovy()
}

在demo/gradle/task目录下创建HelloTask.groovy文件,此文件中:

import org.gradle.api.DefaultTask
import org.gradle.api.tasks.Optional
import org.gradle.api.tasks.TaskAction

class HelloThreeTask extends DefaultTask {
    @Optional
    String message = 'I am 34sir'

    @TaskAction
    def hello() {
        println "hello world $message"
    }
}

查看buildSrc下的目录:

image.png
使用此task:
build.gradle中:
task helloThree(type: HelloThreeTask) {
    message = "hello three"
}

执行命令:gradle helloThree
查看运行结果:

> Task :app:helloThree
hello world hello three

自定义plugin

build.gradle中:

apply plugin: DateAndTimePlugin
dateAndTime {
    timeFormat = 'HH:mm:ss.SSS'
    dateFormat = 'MM/dd/yyyy'
}

// 每一个自定义的Plugin都需要实现Plugin<T>接口
class DateAndTimePlugin implements Plugin<Project> {
    //该接口定义了一个apply()方法,在该方法中,我们可以操作Project,
    //比如向其中加入Task,定义额外的Property等。
    void apply(Project project) {
        println "Current time"
        project.extensions.create("dateAndTime", DateAndTimePluginExtension)
        //每个Gradle的Project都维护了一个ExtenionContainer,
        //我们可以通过project.extentions进行访问
        //比如读取额外的Property和定义额外的Property等。
        project.task('showTime') << {
            println "Current time is " + new Date().format(project.dateAndTime.timeFormat)
        }

        project.tasks.create('showDate') << {
            println "Current date is " + new Date().format(project.dateAndTime.dateFormat)
        }
    }
}
//向Project中定义了一个名为dateAndTime的extension
//并向其中加入了2个Property,分别为timeFormat和dateFormat
class DateAndTimePluginExtension {
    String timeFormat = "MM/dd/yyyyHH:mm:ss.SSS"
    String dateFormat = "yyyy-MM-dd"
}

执行gradle showTime以及gradle showDate查看结果:

> Task :app:showTime
Current time is 14:52:11.539
> Task :app:showDate
Current date is 02/08/2018

执行task

执行多个task

gradle task1 task2 [...]

简化执行

对于名字太长的task可以简化执行
例如上面的hello就可以这样执行:

gradle hell //能唯一标识出task的字符串即可

自动执行任务

不想在每次打包前执行各种自定义的任务,我们可以在build时自动执行任务:

afterEvaluate {
    tasks.matching {
        // 以process开头以ReleaseJavaRes或DebugJavaRes结尾的task
        it.name.startsWith('process') && (it.name.endsWith('ReleaseJavaRes') || it.name.endsWith
                ('DebugJavaRes'))
   }.each { task ->
        task.dependsOn(chVer, chSo)  // 任务依赖:执行task之前需要执行dependsOn指定的任务
    }
}

type

task clean(type: Delete) {  
    delete rootProject.buildDir  
}

以上这段task想必大家都见过,这里出现了一个新的概念:type
task的有很多中类型,此处的类型就是delete,任务是删除文件

常见的type

替换AndroidManifest文件:

task chVer(type: Copy) { // 指定Type为Copy任务
    from "src/main/manifest/AndroidManifestCopy.xml"  // 复制src/main/manifest/目录下的AndroidManifest.xml
    into 'src/main'  // 复制到指定目标目录
    rename { String fileName -> //在复制时重命名文件
        fileName = "AndroidManifest.xml" // 重命名
    }
}

替换so文件:

task chSo(type: Copy) {
    from "src/main/jniLibs/test"   // 复制test文件夹下的所有so文件
    into "src/main/jniLibs/armeabi-v7a" //复制到armeabi-v7a文件夹下
}
task syncDependencies(type: Sync) {
    from 'my/shared/dependencyDir'
    into 'build/deps/compile'
}

// 你可以保护目标目录已经存在的文件。匹配的文件将不会被删除。
task sync(type: Sync) {
    from 'source'
    into 'dest'
    preserve {
        include 'extraDir/**'
        include 'dir1/**'
        exclude 'dir1/extra.txt'
    }
}
task zip(type: Zip) {
    from 'src/dist'
    into('libs') 
}
上一篇 下一篇

猜你喜欢

热点阅读