啥也不说了,直接进入主题吧。本篇文章主要根据实际开发中遇到的需求,讲解使用 Gradle 对应用的不同版本进行个性化定制。

场景介绍

  1. 一般的应用基本上都有正式服和测试服,这个就不需要多说了。但是有些应用可能还有超管服务器专供运营人员使用,对应用内的一些内容进行监管,具有一些管理员才有的操作权限。
  2. 开发过程中发布测试服务器的安装包需要在版本号后面增加版本序号,超管服务器的包在版本号后面增加管理员文字,线上包则正常显示版本号。
  3. 每次打包 versionCode 自增,避免发版时忘记手动修改导致老版本不能覆盖安装。
  4. 超管包的渠道名为 admin,日常运行的 debug 包渠道名为 test,上线的包使用加固软件进行多渠道加固。
  5. debug 包和 release 包使用同样的签名,避免直接运行的 debug 包因为签名问题不能使用需要校验签名的第三方服务,比如:QQ 登录,微信登录,高德地图。
  6. debug 包打印日志信息,release 包不打印日志信息

以上某些场景从我工作以来就一直存在,以前用 eclipse 开发时除了每次都手动去修改一些开关变量也没啥好办法,可能是因为当时菜 ╮(╯▽╰)╭(如果你们有什么好方法的话)。后来切换到 Android Studio 后使用 Gradle 进行依赖管理已经让人很是欣喜,既然如此能不能使用 Gradle将以上问题统统解决,完全自动化呢?答案是:必须的。

解决问题

先上完整的 Gradle 配置。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
android {
compileSdkVersion 25
buildToolsVersion "25.0.3"
defaultConfig {
applicationId "com.imliujun.gradle"
minSdkVersion 16
targetSdkVersion 25
versionCode gitVersionCode() //获取 git 的 commit 次数
versionName rootProject.ext.versionName
 
manifestPlaceholders = [UMENG_APP_KEY : "填你的友盟 APP KEY",
UMENG_CHANNEL_VALUE: "默认的渠道名"]
}
 
signingConfigs {
//在这里配置相关的签名信息
keyStore {
storeFile file("test.jks")
storePassword "111111"
keyAlias "test"
keyPassword "111111"
}
}
 
buildTypes {
release {
// 不显示Log
buildConfigField "boolean", "LOG_DEBUG", "false"
signingConfig signingConfigs.keyStore //设置签名文件
}
 
debug {
// 显示Log
buildConfigField "boolean", "LOG_DEBUG", "true"
versionNameSuffix "-debug" //设置后缀
signingConfig signingConfigs.keyStore //设置签名文件
manifestPlaceholders.UMENG_CHANNEL_VALUE = "test" //修改渠道名
}
}
 
productFlavors {
offline {
buildConfigField "String", "DOMAIN_NAME", "\"https://offline.domain.com/\""
versionName getTestVersionName() //修改 versionName
}
 
online {
buildConfigField "String", "DOMAIN_NAME", "\"https://online.domain.com/\""
}
 
admin {
buildConfigField "String", "DOMAIN_NAME", "\"https://admin.domain.com/\""
versionName rootProject.ext.versionName + "-管理员" //修改 versionName
manifestPlaceholders.UMENG_CHANNEL_VALUE = "admin" //修改渠道名
}
}
}

根目录下的 build.gradle 文件进行如下配置,主要是将版本号和测试包的序号抽取出来:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
ext {
versionName = "2.0.2"
testNum = "0001"
}
 
def getTestVersionName() {
return String.format("%s.%s", rootProject.ext.versionName,
rootProject.ext.testNum)
}
 
static int gitVersionCode() {
def count = "git rev-list HEAD --count".execute().text.trim()
return count.isInteger() ? count.toInteger() : 0
}

下面就根据场景来依次介绍对应的配置代码。

配置服务器版本

这里创建了三个 flavor,分别是 offline 测试服online 正式服admin 超管服。并且通过 buildConfigField 动态配置服务器的 URL 常量值到编译后自动生成的 BuildConfig 类中。

自动生成的 BuildConfig 类

图中可以看到,不止有 DOMAIN_NAME 常量值,还有一个 FLAVOR 常量。这个 FLAVOR 常量中的值是 offline,代表当前在 offline 这个版本上面。那怎么切换到其他的服务器呢?

切换不同的变种版本

点开左下角的 Build Variants, 可以自由切换当前运行的版本。需要在管理员包中开启一些高级的功能,可以判断 FLAVOR 的值是不是 admin,如果是的话就显示管理员的操作布局。当然必不可少的要对用户权限进行校验哦。

定制 versionName

大家看上图 BuildConfig 类中 VERSION_NAME 常量的值为 2.0.2.0001-debug,当前是测试服的 debug 包,所以 versionName 应该是正常的 2.0.2 版本后面拼上当前出包的序号 0001 ,再拼上 debug 的后缀,所以完整的版本号是 2.0.2.0001-debug

看看不同服务器版本的 VERSION_NAME :

  • offlineRelease 版本为 2.0.2.0001
  • offlineDebug 版本为 2.0.2.0001-debug
  • adminRelease 版本为 2.0.2-管理员
  • adminDebug 版本为 2.0.2-管理员-debug
  • onlineRelease 版本为 2.0.2
  • onlineDebug 版本为 2.0.2-debug

如果我们接口需要上传版本号给服务器呢?肯定不能直接上传这些定制化后的 VERSION_NAME,那么我们在 Gradle 中增加一个 buildConfigField 将原始的版本号存起来就好了。

1
buildConfigField "String", "versionNumber", "\"${rootProject.ext.versionName}\""

versionCode 自增

这里采用了主流的方式,使用 git 的 commit 次数作为 versionCode 的值。不用担心这个值会超过 int 的上限,你得敲烂多少键盘才能提交 2147483648次 commit

1
2
3
4
static int gitVersionCode() {
def count = "git rev-list HEAD --count".execute().text.trim()
return count.isInteger() ? count.toInteger() : 0
}

个性化渠道名

Gradle 多渠道打包的文章太多了,相关的基础我就不讲了。简单讲下本文相关的配置吧。

通过定义 manifestPlaceholders 键值对,在 AndroidManifest.xml 文件中使用占位符的方式动态输入 UMENG_APPKEY 和 UMENG_CHANNEL 。

然后在 debug 的 buildType 中修改渠道名为 test,在 admin 的 Flavor 中修改渠道名为 admin。如果选择 adminDebug 版本,则渠道名为 testbuildType 中的配置会覆盖掉 Flavor 中的配置。

由于我们线上使用第三方加固,所以多渠道包就交给第三方加固软件来生成了。

关于多渠道打包我还有两句话要说,以前使用 Gradle 进行多渠道打包,通过代码自定义修改 apk 文件的输出路径,Android Studio 编译的时候时不时的报一些文件找不到的错误,以前都是通过在 Gradle 文件中随便修改一点东西然后刷新一下 Gradle 文件来解决。现在我打包不修改输出路径,再也没遇到以前的那些问题了。

建议大家使用 assemble 命令来进行打包,比如我要出一个测试包使用 ./gradlew assembleOfflineRelease 命令,apk 文件生成在 /build/outputs/apk/ 目录下。直接执行 assemble 命令是编译 Build Variants 中的所有包,如果你要编译指定版本的包,直接在 assemble 命令后面拼上指定的 Build Variant 就好了。

debug 包使用 release 签名

这个问题在 eclipse 时代,可以直接在设置里面配置 debug 签名文件为 release 的签名文件。
用 Android Studio 只需要在 Gradle 中配置就好了。

首先配置签名文件的信息:

1
2
3
4
5
6
7
8
9
signingConfigs {
//在这里配置相关的签名信息
keyStore {
storeFile file("test.jks")
storePassword "111111"
keyAlias "test"
keyPassword "111111"
}
}

然后在 buildTypes 中设置签名信息:

1
2
3
4
5
6
7
8
9
buildTypes {
release {
signingConfig signingConfigs.keyStore //设置签名文件
}
 
debug {
signingConfig signingConfigs.keyStore //设置签名文件
}
}

日志开关

这个太简单了,不想单独列出来。不过上面场景里面提出来了,就简单一行代码展示吧。

1
2
3
4
5
6
7
8
9
release {
// 不显示Log
buildConfigField "boolean", "LOG_DEBUG", "false"
}
 
debug {
// 显示Log
buildConfigField "boolean", "LOG_DEBUG", "true"
}

在日志工具类中使用 BuildConfig 类中的 LOG_DEBUG 常量来判断当前是否应该输出日志。

当然也可以用这个开关来控制开启严格模式等其他只适合在 debug 模式下开启的设置。

结语

唧唧歪歪说了这么多,在懂的人眼中自然很简单,在对 gradle 一点都不了解的人眼中就可以直接复制过去用了。当然我是不建议直接复制,毕竟需求稍微一改,你可能就束手无策了。建议大家还是以理解为主,掌握其原理自然一通百通。

相关阅读

Android Studio 3.0 上 Gradle 改动

使用 Gradle 实现一套代码开发多个应用

使用 Gradle 对应用进行个性化定制的更多相关文章

  1. Webstorm+Webpack+echarts构建个性化定制的数据可视化图表&&两个echarts详细教程(柱状图,南丁格尔图)

    Webstorm+Webpack+echarts   ECharts 特性介绍 ECharts,一个纯 Javascript 的图表库,可以流畅的运行在 PC 和移动设备上,兼容当前绝大部分浏览器(I ...

  2. ECharts-基于Canvas,纯Javascript图表库,提供直观,生动,可交互,可个性化定制的数据可视化图表

    ECharts http://ecomfe.github.com/echarts 基于Canvas,纯Javascript图表库,提供直观,生动,可交互,可个性化定制的数据可视化图表.创新的拖拽重计算 ...

  3. 【SSO单点系列】(2):CAS4.0 登录页的个性化定制

    上一篇 [SSO单点系列](1):CAS环境的搭建介绍了CAS最简单环境的搭建,以及一个例子用来讲解CAS的一个最基础的用法. 今天主要是介绍如何对CAS登录页进行个性化定制.    一.开始 下图是 ...

  4. iTerm2和oh-my-zsh的个性化定制

    终于在某东做活动新入手了一台air,看身边小伙伴的mac的终端感觉好炫酷,于是乎准备自己也捯饬捯饬,google了一下,发现了osx平台上的终端神器iTerm2和用来代替bash的oh-my-zsh, ...

  5. Maven的个性化定制

    用Maven的小伙伴都知道,Maven的宗旨是约定优于配置(Convention Over Configuration). 在宗旨的前提下Maven也提供了个性化定制的Profile,让我们看看使用方 ...

  6. 汽车行业如何个性化定制转型?看APS系统在这家企业的运用

    传统汽车行业中往往采用的是按库存推动式生产,一旦市场产生变动就会造成大量的生产,给企业带来大批的资金压力,而另一方面采用按单生产的方式企业往往面临供应链,产能的诸多约束条件限制,稍有不慎就会带来产线停 ...

  7. CRM系统个性化定制的对企业的优势作用

    伴随着科学技术的不断发展,企业信息化建设也在持续地开展.企业管理模式已经开始由传统模式向信息化转变,并且越来越多的企业开始使用互联网软件来进行辅助管理,这一趋势也让CRM客户管理系统得到快速的发展.市 ...

  8. QMenu的个性化定制

    经常使用菜单,菜单的定制相当重要,普通的样式设置不难,一般需求足以实现(QMenu + QAction).如果要足够个性,则需要进行一定的定制.     说起定制,其实也是利用Qt中现成的组件进行组装 ...

  9. 个性化定制——物流app

    众所周知,在互联网不断迈进的大环境下,各行各业都不免在这大潮下纷纷卷入.人们早已不再满足于传统行业,即便是所谓的新兴行业所带来的体验,他们更多的希望能够在便捷的基础上获取更加个性化的服务,个性化服务在 ...

随机推荐

  1. Linux命令学习chroot和chmode

    chroot:chang root http://man.linuxde.net/chroot https://baike.baidu.com/item/chroot 1.限制被CHROOT的使用者所 ...

  2. Windows下PyInstaller的使用教程

    直接使用Python开发的软件时有许多不方便的地方,如需要安装特定的Python环境,需要安装依赖库.为了便于部署,需要将Python源代码编译成可执行文件,编译后的可执行文件就能脱离python环境 ...

  3. delphi clientdataset判断某一行值是否存在

    ClientDataSet的数据查找.我所介绍的心得和技巧都是用ClientDataSet来做范例,也可以应用于其他的一些DataSet.废话就不多说了.我们还是先看代码,让后再总结.1.Scanni ...

  4. 用send_keys输入文本的方法

    我们使用app时,输入文字都是调用软键盘.在自动化测试中当然也可以调用软键盘,但是由于输入法设计上的差异,有时候不能达到很好的效果. 例如,搜狗拼音输入法: 选择4-咖啡,然而多打几次,输入法就把“咖 ...

  5. 一本通1633【例 3】Sumdiv

    1633:[例 3]Sumdiv 时间限制: 1000 ms         内存限制: 524288 KB [题目描述] 原题来自:Romania OI 2002 求 ABAB 的所有约数之和 mo ...

  6. java 枚举 封装操作方法

    前言: 由于刚转java项目,所以对于java语言比较陌生,尤其是lambda和一些诸如(一个java文件只能有一个public class)等等的零散知识点之类... 使我觉得就语言的层级来说..n ...

  7. MT【103】二阶递推找规律

    评:如果直接找$a_n$的二阶递推式:$a_{n+2}-2\sqrt{2}a_{n+1}-a_n=0$有根号,不利于估计尾数.

  8. 【BZOJ1449】[JSOI2009]球队收益(网络流,费用流)

    [BZOJ1449][JSOI2009]球队收益(网络流,费用流) 题面 BZOJ 洛谷 题解 首先对于一支队伍而言,总共进行多少场比赛显然是已知的,假设是\(n_i\)场,那么它的贡献是:\(C_i ...

  9. NOI2018旅游记

    这居然是我第一次参加非NOIP的NOI系列赛事,有点小期待啊 前几天的UNR我暴露出了许多问题,而且翻了好多分,不过令人震惊的是假设Day1不停电(导致已经写好的T3没交上去)我居然有rk10,虽然并 ...

  10. WEB入门之十一 JS面向对象

    学习内容 JavaScript类的定义模式 JavaScript继承的实现 JavaScript抽象类 JavaScript解析XML 能力目标 深入了解JavaScript类的定义模式 理解Java ...