使用 Gradle 对应用进行个性化定制

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

场景介绍

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

以上某些场景从我工作以来就一直存在,以前用 eclipse 开发时除了每次都手动去修改一些开关变量也没啥好办法,可能是因为当时菜 ╮(╯▽╰)╭(如果你们有什么好方法的话)。

后来切换到 Android Studio 后使用 Gradle 进行依赖管理已经让人很是欣喜,既然如此能不能使用 Gradle 将以上问题统统解决,完全自动化呢?答案是:必须的。

解决问题

先上完整的 Gradle 配置。

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 文件进行如下配置,主要是将版本号和测试包的序号抽取出来:

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
}

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

1. 配置服务器版本

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

自动生成的 BuildConfig 类
自动生成的 BuildConfig 类

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

切换不同的变种版本
切换不同的变种版本

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

2. 定制 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 将原始的版本号存起来就好了。

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

3. versionCode 自增

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

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

4. 个性化渠道名

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

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

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

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

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

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

5. debug 包使用 release 签名

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

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

signingConfigs {
    //在这里配置相关的签名信息
     keyStore {
         storeFile file("test.jks")
         storePassword "111111"
         keyAlias "test"
         keyPassword "111111"
     }
}

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

buildTypes {
    release {
        signingConfig signingConfigs.keyStore //设置签名文件
    }

    debug {
        signingConfig signingConfigs.keyStore  //设置签名文件
    }
}

6. 日志开关

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

release {
    // 不显示Log
    buildConfigField "boolean", "LOG_DEBUG", "false"
}

debug {
    // 显示Log
    buildConfigField "boolean", "LOG_DEBUG", "true"
}

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

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

结语

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

欢迎关注微信公众号:**大脑好饿**,更多干货等你来尝
欢迎关注微信公众号:**大脑好饿**,更多干货等你来尝
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,590评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,808评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,151评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,779评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,773评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,656评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,022评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,678评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,038评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,756评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,411评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,005评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,973评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,053评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,495评论 2 343

推荐阅读更多精彩内容