androidgradle开发指南

wx5aae83353cec4 wx5aae83353cec4     2022-12-02     217

关键词:


Gradle简介

Gradle 是一个优秀的构建系统和构建工具,它允许通过插件创建自定义的构建逻辑。它具有如下一些特点:

  • 采用了 Domain Specific Language(DSL 语言) 来描述和控制构建逻辑。
  • 构建文件基于 Groovy,并且允许通过混合声明 DSL 元素和使用代码来控制 DSL 元素以控制自定义的构建逻辑。
  • 支持 Maven 或者 Ivy 的依赖管理。
  • 非常灵活。允许使用最好的实现,但是不会强制实现的方式。
  • 插件可以提供自己的 DSL 和 API 以供构建文件使用。
  • 良好的 API 工具供 IDE 集成。

使用Gradle方式来构建项目,主要是为了达到如下的一些目的:

  • 让重用代码和资源变得更加容易
  • 让创建同一应用程序的不同版本变得更加容易,无论是多个 apk 发布版本还是同一个应用的不同定制版本
  • 让构建过程变得更加容易配置,扩展和定制。
  • 整合优秀的 IDE。

构建项目基础

文件构建

一个Gradle项目的构建过程定义在build.gradle文件中,位于项目的根目录下。一个最简单的Gradle纯Java项目的build.gradle文件包含以下内容。

apply plugin: java

上面的代码是引入了Gradle的Java插件,这个插件提供了所有构建和测试Java应用程序所需要的东西。例如,下面是一个最简单的Android项目的build.gradle文件的源代码。

buildscript 
repositories
google()
jcenter()


dependencies
classpath com.android.tools.build:gradle:3.4.1



allprojects
repositories
google()
jcenter()




task clean(type: Delete)
delete rootProject.buildDir

build.gradle文件

在一个Android项目中一般会出现至少2个build.gradle文件,一个是project的gradle文件,一个是app module的gradle文件。

gradle文件中会涉及很多的常用命令和代码,它们的具体含义如下:

1,jcenter()

代码托管库,设置后可以在项目中引用jcenter上的开源项目,声明在build.gradle文件的repositories闭包中。

2,gradle插件及版本号

经常会看到如下一段代码:

 classpath com.android.tools.build:gradle:3.4.1

3,Android 闭包配置

在build.gradle的Android闭包中会看到一些常用的配置,如下所示:

  • compileSdkVersion: 用于指定项目的编译版本。
  • buildToolsVersion: 用于指定项目的构建工具的版本。
  • applicationId: 用于指定项目的包名,在创建项目的时候已经指定了包名,当要修改整个项目的包名时可以在此更改。
  • minSdkVersion: 项目最低的兼容版本。
  • targetSdkVersion:表示你在该目标版本上已经做过了充分的测试,系统将为你的应用开启一些最新的功能和特性。假如targetSdkVersion 为23或者更高,那么在Android6.0中运行这个应用时会开启新的功能和特性;若设置成了22,只能说明你的应用程序在Android5.1系统上做过了充分的测试,Android6.0的新功能就不会启用。
  • versionCode: 项目的版本号。
  • versionName: 项目版本号的版本名。

4,buildTypes闭包

此配置包一般会包含两个闭包配置,一个是debug,一个release;当然也可以有其他闭包。debug闭包用于生成测试版安装文件的配置,release闭包用于生成正式版安装文件的配置。此文件的配置如下:

  • minifyEnabled:用于设置是否对项目的代码进行混淆。true代表开启,false代表关闭。
  • proguardFiles:指定混淆时使用的文件。
  • proguard-android.txt:在Android SDK 目录下的,里面是所有项目通用的混淆规则。
  • proguard-rules.pro:是在当前项目根目录下的,里面编写当前项目特有的混淆规则。

5,dependencies闭包

Android Studio 项目开发中一共有三种依赖方式:本地依赖、库依赖和远程依赖。

  • 本地依赖:可以对本地的Jar包或目录添加依赖关系。
  • 库依赖:可以对项目中的库模块添加依赖关系。
  • 远程依赖:=可以对jcenter库上的开源项目添加依赖关系。

项目结构

Gradle遵循约定优先于配置的概念,在可能的情况尽可能提供合理的默认配置参数。Android基本的项目开始于两个名为“source sets”的组件,即main source code和test code。它们分别位于:src/main/和src/androidTest/文件中。
对于Java plugin和Android plugin来说,它们的Java源代码和资源文件路径如下:java/和resources/文件目录中。

对于Android plugin来说,它还拥有以下特有的文件和文件夹结构:

  • AndroidManifest.xml
  • res/
  • assets/
  • aidl/
  • rs/
  • jni/

配置结构

当默认的项目结构不适用的时候,你可能需要去自定义配置它。根据Gradle文档,重新为Java项目配置sourceSets可以使用以下方法:

sourceSets 
main
java
srcDir src/java

resources
srcDir src/resources


当然,也可以使用如下的配置方式:

sourceSets 
main.java.srcDirs = [src/java]
main.resources.srcDirs = [src/resources]

Android Plugin使用的是类似的语法。但是由于它使用的是自己的sourceSets,这些配置将会被添加在android对象中。

以下是一个示例,它使用了旧项目结构中的main源码,并且将androidTestsourceSet组件重新映射到tests文件夹。

android 
sourceSets
main
manifest.srcFile AndroidManifest.xml
java.srcDirs = [src]
resources.srcDirs = [src]
aidl.srcDirs = [src]
renderscript.srcDirs = [src]
res.srcDirs = [res]
assets.srcDirs = [assets]


androidTest.setRoot(tests)

构建任务

通用任务

添加一个插件到构建文件中将会自动创建一系列构建任务(build tasks)去执行(注:gradle属于任务驱动型构建工具,它的构建过程是基于Task的)。Java plugin和Android plugin都可以创建以下task。

  • assemble:这个task将会组合项目的所有输出。
  • check:这个task将会执行所有检查操作。
  • build:这个task将会执行assemble和check两个task的所有工作。
  • clean:这个task将会清空项目的输出。

实际上assemble,check,build这三个task不做任何事情,它们只是一个Task标志,用来告诉android plugin添加实际需要执行的task,执行某些具体的工作。

例如,添加了findbugs plugin将会创建一个新的task并且让check task依赖于这个新的task。当check task被调用的时候,这个新的task将会先被调用。

在命令行环境中,可以执行以下命令来获取更多高级别的task。

gradle tasks

查看所有task列表和它们之间的依赖关系,可以执行以下命令。

gradle tasks --all

需要说明的是,Gradle会自动监视一个task声明的所有输入和输出。两次执行build task并且期间项目没有任何改动,gradle将会使用UP-TO-DATE通知所有task。这意味着第二次build执行的时候不会请求任何task执行。这允许task之间互相依赖,而不会导致不需要的构建请求被执行。

Java 项目的Task

Java plugin主要创建了两个task,一个是assemble,一个是check,它们都依赖于main task(一个标识性的task)。通常情况下,你只需要调用到assemble和check,不需要其他task。

Android Task

Android plugin使用相同的约定以兼容其他插件,并且附加了自己的标识性task,常用的Task包括:

  • assemble:这个task用于组合项目中的所有输出。
  • check:这个task用于执行所有检查。
  • connectedCheck:这个task将会在一个指定的设备或者模拟器上执行检查,它们可以同时在所有连接的设备上执行。
  • deviceCheck:通过APIs连接远程设备来执行检查,这是在CL服务器上使用的。
  • build:这个task执行assemble和check的所有工作。
  • clean:这个task清空项目的所有输出。

这些新的标识性task是必须的,以保证能够在没有设备连接的情况下执行定期检查。注意build task不依赖于deviceCheck或者connectedCheck。

一个Android项目至少拥有两个输出:debug APK和release APK,每一个输出都拥有自己的标识性task以便能够单独构建它们。

  • assemble
  • assembleDebug
  • assembleRelease

其中assemble task依赖于其他两个task,所以执行assemble将会同时构建出两个APK。gradle在命令行终端上支持骆驼命名法的task简称,例如,执行

gradle aR

命令等同于执行

gradle assembleRelease

同时,check task也可以拥有自己的依赖,常见的依赖有:

  • check
  • lint
  • connectedCheck
  • connectedAndroidTest
  • connectedUiAutomatorTest(目前还没有应用到)
  • deviceCheck

Android 构建

Android plugin提供了大量DSL用于构建系统定制。

Manifest 属性

通过SDL,我们可以配置一下manifest选项。

  • minSdkVersion
  • targetSdkVersion
  • versionName
  • applicationId
  • package Name for the test application
  • Instrumentation test runner

例如:

android 
compileSdkVersion 29
buildToolsVersion "29.0.2"

defaultConfig
applicationId "com.xzh.gradleplugin"
minSdkVersion 15
targetSdkVersion 29
versionCode 1
versionName "1.0"
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"


之前的Android Plugin版本使用packageName来配置manifest文件中的packageName属性。从0.11.0版本开始,你需要在build.gradle文件中使用applicationId来配置manifest文件中的packageName属性。这是为了消除应用程序的packageName(也是程序的ID)和java包名所引起的混乱。

在构建文件中定义的强大之处在于它是动态的。例如,可以从一个文件中或者其它自定义的逻辑代码中读取版本信息:

def computeVersionName() 
...


android
compileSdkVersion 29
buildToolsVersion "29.0.2"

defaultConfig
applicationId "com.xzh.gradleplugin"
minSdkVersion 15
targetSdkVersion 29
versionCode 1
versionName computeVersionName()
testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"


如果一个属性没有使用DSL进行设置,一些默认的属性值将会被使用。以下表格是可能使用到的值:

Property Name

Default value in DSL object

Default value

versionCode

-1

value from manifest if present

versionName

null

value from manifest if present

minSdkVersion

-1

value from manifest if present

targetSdkVersion

-1

value from manifest if present

applicationId

null

value from manifest if present

testApplicationId

null

applicationId + “.test”

testInstrumentationRunner

null

android.test.InstrumentationTestRunner

signingConfig

null

null

proguardFile

N/A (set only)

N/A (set only)

proguardFiles

N/A (set only)

N/A (set only)

构建类型

默认情况下,Android Plugin会自动给项目设置同时构建应用程序的debug和release版本。两个版本之间的不同主要围绕着能否在一个安全设备上调试,以及APK如何签名。

Debug版本采用使用通用的name/password键值对自动创建的数字证书进行签名,以防止构建过程中出现请求信息。Release版本在构建过程中没有签名,需要稍后再签名。

这些配置通过一个BuildType对象来配置。默认情况下,这两个实例都会被创建,分别是一个debug版本和一个release版本。

Android plugin允许像创建其他构建类型一样定制debug和release实例。这需要在buildTypes的DSL容器中配置如下选项。

android 
buildTypes
debug
applicationIdSuffix ".debug"


jnidebug.initWith(buildTypes.debug)
jnidebug
packageNameSuffix ".jnidebug"
jnidebugBuild true


以上代码片段完成了如下功能:

  • 配置默认的debug构建类型。
  • 将debug版本的包名设置为.debug以便能够同时在一台设备上安装debug和release版本的apk。
  • 创建了一个名为jnidebug的新构建类型,并且这个构建类型是debug构建类型的一个副本。
  • 继续配置jnidebug构建类型,允许使用JNI组件,并且也添加了不一样的包名后缀。

创建一个新的构建类型就是简单的在buildType标签下添加一个新的元素,并且可以使用initWith()或者直接使用闭包来配置它。以下是一些可能使用到的属性和默认值:

Property Name

Default values for debug

Default values for release / other

debuggable

true

false

jniDebugBuild

false

false

renderscriptDebugBuild

false

false

renderscriptOptimLevel

3

3

applicationIdSuffix

null

null

versionNameSuffix

null

null

signingConfig

android.signingConfigs.debug

null

zipAlign

false

true

runProguard

false

false

proguardFile

N/A (set only)

N/A (set only)

proguardFiles

N/A (set only)

N/A (set only)

除了以上属性之外,Build Type还会受项目源码和资源影响:对于每一个Build Type都会自动创建一个匹配的sourceSet。默认的路径为:

  src/<buildtypename>/

这意味着BuildType名称不能是main或者androidTest(因为这两个是由plugin强制实现的),并且他们互相之间都必须是唯一的。跟其他sourceSet设置一样,Build Type的source set路径可以重新被定向。

android 
sourceSets.jnidebug.setRoot(foo/jnidebug)


androidgradle插件开发

GradleGradle简介Gradle是一个基于ApacheAnt和ApacheMaven概念的项目自动化构建开源工具,支持maven、Ivy仓库,支持传递性依赖管理,而不需要远程仓库或者是pom.xml和ivy.xml配置文件。Gradle使用一种基于Groovy的特定领域语言(DSL)来声明项目... 查看详情

androidgradle完整指南

为什么需要学Gradle?Gradle是Android现在主流的编译工具,虽然在Gradle出现之前和之后都有对应更快的编译工具出现,但是Gradle的优势就在于它是亲儿子,Gradle确实比较慢,这和它的编译过程有关,但是现在的Gradle编译速度已经有了... 查看详情

《gradle权威指南》--androidgradle多渠道构建(代码片段)

No1:BuildVariant=BuildType+ProductFlavorBuildVariant:构建的产物BuildType:构建的类型ProductFlavor:构建的渠道androidcompileSdkVersion23buildToolsVersion"23.0.1"productFlavorsgooglebaiduNo2:Flurry多渠道和友盟多渠道构建Flu 查看详情

自定义androidgradle--启动zipalign优化

参考技术A----《AndroidGradle权威指南》学习笔记总目录:Gradle学习系列--目录上一篇:自定义AndroidGradle--使用混淆 查看详情

androidgradle插件开发基础(代码片段)

什么是Gradle在Gradle官方文档上是这么描述的:Gradle是一种开源构建自动化工具,其设计足够灵活,几乎可以构建任何类型的软件。Gradle允许您构建任何软件,因为它对您尝试构建的内容或应该如何完成几乎没有做... 查看详情

androidgradle插件开发基础(代码片段)

什么是Gradle在Gradle官方文档上是这么描述的:Gradle是一种开源构建自动化工具,其设计足够灵活,几乎可以构建任何类型的软件。Gradle允许您构建任何软件,因为它对您尝试构建的内容或应该如何完成几乎没有做... 查看详情

androidgradle插件开发基础(代码片段)

什么是Gradle在Gradle官方文档上是这么描述的:Gradle是一种开源构建自动化工具,其设计足够灵活,几乎可以构建任何类型的软件。Gradle允许您构建任何软件,因为它对您尝试构建的内容或应该如何完成几乎没有做... 查看详情

androidgradle中的transform(代码片段)

前言         逐步整理的一系列的总结:     AndroidGradle插件开发初次交手(一)        AndroidGradle的基本概念梳理(二)       Android自定义Gradle插件的完整流程(三)       Android自... 查看详情

androidgradle的基本概念梳理(代码片段)

...        逐步整理的一系列的总结:        AndroidGradle插件开发初次交手(一)        AndroidGradle的基本概念梳理(二)        Android自定义Gradle插件的完整流程(三)   Android自定义Ta... 查看详情

androidgradle学习笔记语言和命令(代码片段)

Gradle支持使用GroovyDSL或KotlinDSL来编写脚本。所以在学习具体怎么写脚本时,我们肯定会考虑到底是使用Kotlin来写还是Groovy来写。不一定说你是KotlinAndroid开发者就一定要用Kotlin来写Gradle,我们得判断哪种写法更适合项目、... 查看详情

项目指南

...的知识。1.项目指南1.1项目结构    新项目应遵循的AndroidGradle项目结构上定义AndroidGradle插件用户指南。Theribot样板项目从开始就是一个很好的参考。1.2 文件的命名1.2.1 类文件    类名称是用UpperCame 查看详情

androidgradle之java字节码(代码片段)

前言       逐步整理的一系列的总结:     AndroidGradle插件开发初次交手(一)        AndroidGradle的基本概念梳理(二)       Android自定义Gradle插件的完整流程(三)       Android自定... 查看详情

androidgradle中的字节码插桩之asm(代码片段)

前言         逐步整理的一系列的总结:     AndroidGradle插件开发初次交手(一)        AndroidGradle的基本概念梳理(二)       Android自定义Gradle插件的完整流程(三)       Android自... 查看详情

androidgradle定义boolean类型(代码片段)

gradle是一种构建工具。在安卓开发中,是利用gradle进行配置的。项目路径如下:我们在开发过程中可以在build.gradle中定义一些配置变量,然后再java中通过此变量来处理不同的业务逻辑。例如,我们想要通过一个is_... 查看详情

androidgradle定义boolean类型(代码片段)

gradle是一种构建工具。在安卓开发中,是利用gradle进行配置的。项目路径如下:我们在开发过程中可以在build.gradle中定义一些配置变量,然后再java中通过此变量来处理不同的业务逻辑。例如,我们想要通过一个is_... 查看详情

androidgradle动态添加模块依赖

参考技术AAndroid开发过程,可能会遇到依赖模块太多,手动的添加修改依赖就会觉得有点麻烦,这个时候可以考虑使用动态添加模块依赖,也是适用像Jenkins自动打包构建,就不需要频繁的去修改模块依赖,提高构建效率。1、工... 查看详情

androidgradle中的实例之动态修改androidmanifest文件(代码片段)

前言        逐步整理的一系列的总结:     AndroidGradle插件开发初次交手(一)        AndroidGradle的基本概念梳理(二)       Android自定义Gradle插件的完整流程(三)       Android自... 查看详情

androidgradle中的实例之动态修改androidmanifest文件(代码片段)

前言        逐步整理的一系列的总结:     AndroidGradle插件开发初次交手(一)        AndroidGradle的基本概念梳理(二)       Android自定义Gradle插件的完整流程(三)       Android自... 查看详情