Gradle筑基篇(五)-Gradle自定义插件实战

简介: 前面几篇文章笔者对Gradle的一些基础认知,groovy基础语法,以及Gradle 项目中常用的一些api进行了讲解。今天笔者再来讲解一些关于`Gradle插件`的使用

携手创作,共同成长!这是我参与「掘金日新计划 · 8 月更文挑战」的第4天,点击查看活动详情 >>

🔥 Hi,我是小余。

本文已收录到 GitHub · Androider-Planet 中。这里有 Android 进阶成长知识体系,关注公众号 [小余的自习室] ,在成功的路上不迷路!

前言

前面几篇文章笔者对Gradle的一些基础认知,groovy基础语法,以及Gradle 项目中常用的一些api进行了讲解。

Gradle系列文章如下:

Gradle筑基篇:
Gradle进阶篇
  • Gradle进阶篇(六)-AGP详解

今天笔者再来讲解一些关于Gradle插件的使用

1.定义

首先来讲下GradleGradle插件有啥区别?

Gradle是一套构建工具,其内部构建过程主要是以Project组成一个树形的生态系统,整个构建流程有自己的生命周期。每个Project又是由若干个Task组成。

Gradle插件你可以理解为是运行在Gradle这套构建系统上的单个task,
执行脚本的编写字节码插庄等,都可以依靠Gradle插件实现。

我们常用的Android Gradle Plugin也是一个Gradle插件模块:

应用插件的ID:‘com.android.application’
或者lib库:‘com.android.library’

2.有哪些优势

  • 1.逻辑复用:Gradle插件将一个公共模块单独的抽离出来,然后上传到共享平台,供其他项目使用
  • 2.插件配置扩展:Gradle插件声明插件扩展,将插件内部参数暴露给对应的Project进行配置,大大提高了插件的可扩展性。

3.插件的形式

  • 1.build script:直接在build.gradle构建脚本中创建对应的插件,这种方式只对当前Project有效,不支持对外提供调用,无复用性,一般不推荐使用
  • 2.buildSrc模块:这种方式是编译器提供的特殊模块,编译器可以自动识别该模块的,对所有的Project可见。但是在项目外不可见,无法给其他工程使用,复用性差。
  • 3.独立插件项目:替插件创建一个单独的项目,这个项目可以单独的打包成一个jar包,然后发布到企业服务器上供其他项目使用,通常这个插件中包含了一个或多个任务的组合,实现具体的插件功能

4.自定义插件实战

下面我会以第三种形式来大家实现一个简单的Gradle插件功能:

需求如下:

在编译过程中实现:将当前编译的版本信息发布到公司服务器上,可以在本地服务器上实时查看编译的版本日志,通过日志的分析可以对当前编译版本进行优化。

步骤如下

  • 1.初始化插件模块目录结构
  • 2.创建插件实现类
  • 3.创建插件扩展Bean
  • 4.创建插件实现的任务:上传版本信息
  • 5.将插件扩展和插件任务集成到Project生命周期中
  • 6.插件发布
  • 7.插件引入

步骤1.初始化插件模块目录结构

首先创建一个Java or Kotlin Library的模块,

步骤1创建.png
在创建的模块build.gradle中引入:

plugins {
    id 'groovy' // Groovy Language
    //id 'org.jetbrains.kotlin.jvm' // Kotlin 
    id 'java-gradle-plugin' // Java Gradle Plugin
}
  • groovy:使用groovy语言开发
  • org.jetbrains.kotlin.jvm:使用kotlin开发引入kotlin核心插件库
  • java-gradle-plugin:Gradle插件的一个辅助插件,可以在我们build目录下自动生成资源属性

设置sourceSets

sourceSets {
    main {
        groovy {
            srcDir 'src/main/groovy'
        }
        resources {
            srcDir 'src/main/resources'
        }
    }
}

工程目录结构如下:

插件目录结构.png

步骤2.创建插件实现类

class UploadVersionPlugin implements Plugin<Project>{
    @Override
    void apply(Project project) {
            println "begin:now this is a ${project.name} 's upload plugin"
    }
}

步骤3.创建插件扩展Bean

class VersionInfo {
    //版本名称
    String versionName
    //版本代码
    int versionCode
    //版本更新信息
    String versionUpdateInfo
}

步骤4.创建插件实现的任务:上传版本信息

class UploadTask extends DefaultTask{
    String url = 'http://127.0.0.1/api/v3/upload/version'
    @TaskAction
    void upload(){
        //1.获取版本信息
        def version = getCurrentVersion()
        //2.发送版本信息
        def response = sendAndReceive(version)
        //3.处理响应:将版本信息以及响应写入到本地文件中
//        checkResponse(response)

    }
    //1.获取版本信息
    def getCurrentVersion(){
        def name = project.extensions.versionInfo.versionName
        def code = project.extensions.versionInfo.versionCode
        def info = project.extensions.versionInfo.versionUpdateInfo
        println "name:$name code:$code info:$info"
        return new VersionInfo(versionName: name,
                     versionCode: code,
                     versionUpdateInfo: info)
    }
    //2.发送版本信息
    void sendAndReceive(VersionInfo version){
        OkHttpClient client = new OkHttpClient()
        FormBody body = new FormBody.Builder()
                .add('versionName',version.versionName)
                .add('versionCode',""+version.versionCode)
                .add('versionUpdateInfo',version.versionUpdateInfo)
                .build()
        Request.Builder builder = new Request.Builder()
                .url(url)
                .post(body)

        def call1 = client.newCall(builder.build())
        call1.enqueue(new Callback() {
            @Override
            void onFailure(@NotNull Call call, @NotNull IOException e) {
                println "push version fail:reason:"+e.message
            }

            @Override
            void onResponse(@NotNull Call call, @NotNull Response response) throws IOException {
                checkResponse(response);
            }
        })
    }
    //3.处理响应:将版本信息以及响应写入到本地文件中
    void  checkResponse(response){
        println "response:"+new String(response.body().bytes())
        
    }
}

记住,在需要执行的方法上面添加TaskAction注解:在我们任务执行的时候就会执行到这个方法。

步骤5.将插件扩展和插件任务集成到Project生命周期中

@Override
void apply(Project project) {
    println "begin:now this is a ${project.name} 's upload plugin"
    //1.在插件中引入extensions中的字段,就是我们Project中配置的扩展字段
    project.extensions.create(EXTENSIVE,VersionInfo.class)
    //2.创建待处理的Task
    project.tasks.create(TASK_NAME,UploadTask.class)
    //3.将uploadTask任务挂架到Project的生命周期中
    def build = project.tasks.getByName('clean')
    def uploadTask = project.tasks.getByName(TASK_NAME)
    //这里使用dependsOn强依赖任务关系
    build.dependsOn(uploadTask)
}

步骤6.插件发布

笔者为了测试,将jar包只发布在本地,测试使用。

使用如下方式发布:

gradlePlugin {
    plugins {
        modularPlugin {
                id = 'com.yuhb.upload'
                implementationClass = 'com.yuhb.upload.UploadVersionPlugin'
        }
    }
}

这个配置在build后自动生成resources文件:这个插件扩展配置是引入的:java-gradle-plugin中。

resources文件自动生成.png
当然也可以直接在resources文件夹中上手动写入该文件

在插件的build.gradle实现下面的逻辑:

uploadArchives {
    repositories {
            mavenDeployer {
                    repository(url:uri('D:/maven_local'))
                    pom.groupId = 'com.yuhb.upload'
                    pom.artifactId = 'uploader'
                    pom.version = '1.0.0'
            }
    }
}

在命令行执行:

./gradlew :uploadversion:uploadArchives    

然后去本地文件夹下面看看是否上传成功:

本地文件成功.png

这里要说明下:

一般情况下都会将自定义插件发布到maven私服或者中央仓库,才可以供其他项目使用
关于如何发布到maven私服,可以查看这篇文章
后期也会出一期文章教大家如何将数据发布到中央仓库

步骤7.插件引入

  • 步骤1:在工程的根build.gradle文件中引入:
buildscript {
        repositories {
                ...
                maven {
                        url uri('D:/maven_local')
                }
        }
        dependencies {
                ...
                classpath 'com.yuhb.upload:upload:1.0.0'
        }
}

说明:

com.yuhb.upload:uploader:1.0.0格式:

引入字段 发布字段
com.yuhb.upload pom.groupId
uploader pom.artifactId
1.0.0 pom.version
  • 步骤2:在子Project中引入插件:
apply plugin: 'com.yuhb.upload'
  • 步骤3:配置extensive插件扩展:
versionInfo {
    versionName = '1.0.0'
    versionCode = 1
    versionUpdateInfo = '当前是第一个版本:初始apk'
}

这个versionInfo扩展是怎么来的呢?

我们看下之前我们配置插件的时候,使用了:

EXTENSIVE = 'versionInfo'
project.extensions.create(EXTENSIVE,VersionInfo.class)

在插件中引入extensions中的字段,就是我们Project中配置的扩展字段:

versionInfo {
    versionName = '1.0.0'
    versionCode = 1
    versionUpdateInfo = '当前是第一个版本:初始apk'
}

就是这里,如果外部配置了versionInfo的扩展字段,就会通过project.extensions获取到,并将数据写入project.extensionsversionInfo属性中:之后就可以使用project.extensionsversionInfo属性访问外部传入的配置数据:

def name = project.extensions.versionInfo.versionName
def code = project.extensions.versionInfo.versionCode
def info = project.extensions.versionInfo.versionUpdateInfo
  • 步骤4:运行rootbuild 任务查看编译信息:
./gradlew build
结果:
> Task :app:uploadTask
name:1.0.0 code:1 info:当前是第一个版本:初始apk


这里运行build可以执行插件中的任务是因为前面笔者将插件Task挂接到了build任务之前:
挂接代码

//3.将uploadTask任务挂架到Project的生命周期中
def build = project.tasks.getByName('build')
def uploadTask = project.tasks.getByName(TASK_NAME)
//这里使用dependsOn强依赖任务关系
build.dependsOn(uploadTask)
    

项目Demo完整代码已经上传Github:
https://github.com/ByteYuhb/a_gradle_plugin_sample

5.总结

本文主要针对我们自定义插件定义以及优势做了一些说明,且使用一个实战项目对自定义插件制作,发布,引入流程做了一个详细的讲解
,Gradle插件部分还有Gradle的上传流程和AGP插件讲解没有讲,后面都会陆续推出。

参考资料

_官网文档

相关文章
|
Java 开发工具 Android开发
Cocos Creator 2.4.6 Android Gradle 版本升级为 6.5.1(插件4.1.0)
Cocos Creator 2.4.6 Android Gradle 版本升级为 6.5.1(插件4.1.0)
329 1
IDEA 降级gradle插件教程
IDEA 降级gradle插件教程
1384 0
IDEA 降级gradle插件教程
|
Android开发
Android Studio中修改gradle插件版本和Gradle版本
Android项目中,我们一般要设置gradle插件版本和gradle版本。 项目根目录下的build.gradle文件中,通过classpath可以指定gradle插件的版本。
|
6月前
|
Java API 项目管理
Java一分钟之-Gradle插件开发:自定义构建逻辑
【6月更文挑战第5天】Gradle插件开发详解:从入门到发布。文章介绍如何创建自定义插件,强调依赖管理、任务命名和配置阶段的理解。示例代码展示插件实现及避免常见问题的方法。最后,讨论插件的发布与共享,助你提升构建效率并贡献于开发者社区。动手实践,打造强大Gradle插件!
137 3
|
7月前
|
存储 缓存 Java
Gradle笔记 八 Gradle 插件(二)
Gradle笔记 八 Gradle 插件
156 0
|
7月前
|
Java
Gradle笔记 八 Gradle 插件(一)
Gradle笔记 八 Gradle 插件
156 0
|
IDE Java 应用服务中间件
以Gradle插件的方式为Java web项目启动Tomcat
在社区版IntelliJ IDEA除了用SmartTomcat,还有什么方式可以在可调试的情况下启动Tomcat呢,来试试com.bmuschko.tomcat插件吧
487 0
以Gradle插件的方式为Java web项目启动Tomcat
|
Java 程序员 Maven
在AndroidStudio中自定义Gradle插件
在AndroidStudio中自定义Gradle插件
在AndroidStudio中自定义Gradle插件
Gradle进阶篇(七)-AGP插件详解
前面几篇文章我们讲解了关于关于`Gradle的基础`,`Gradle生命周期`,`Gradle相关Api`的讲解,以及`Gradle自定义插件`,`Gradle Maven仓库管理`.今天这篇文章我们来讲解下`Android Gradle Plugin`相关知识。
|
存储 Java API
Gradle筑基篇(六)-使用Maven实现组件化类库发布
前面几篇文章我们讲解了关于关于`Gradle的基础`,`Gradle生命周期`,`Gradle相关Api`的讲解,以及`Gradle自定义插件`,这篇文章我们就来讲解下如何使用`Maven进行组件化发布`