谈谈Android开发中的Gradle那些事之不同BuildType编译出不同版本号的apk

今天我们要来谈谈 如何让不同的BuildType编译出不同的版本号

  • 没搞错吧?
  • 这有什么用??
  • 为什么会有这样的需求???
不想当产品的QA不是一个好RD

RD不只是PM的RD, QA也总能在非常时期提出一些非常独(坑)特(爹)的需求,而我们只能说没问题!(: 男人怎么能说自己不行

废话不多说,正式进入今天的主题:

为什么要编译出不同的版本号呢?
相信大部分公司为了更好的统计、分析app线上crash率等,都会用到一些第三方crash统计,这里拿Fabric举例
Fabric 只能根据app的versionCode值(也就是最终写到manifest文件中的versionCode值)来区分app版本。
这里我为什么说最终呢,别着急,耐心看下去。

此时,我们的最(罪 )佳(魁)P(祸)M(首) QA发话了..

那个 ...
考虑到QA在测试的过程中会遇到一些难以复现的crash等,我们需要在测试环境下也统计crash信息
且 ...
要和线上的区分开来

于是我们的故事就从这里开始了......

如果说要根据不同的Flavor编译出不同的版本号,那太easy了
(这里我们的versionCode是根据versionName按规则转换而来的,转换规则不足3位前面补0: 这里versionName 1.2.1)

productFlavors {  
// 测试环境的包 
 dev { 
   versionCode 121
  } 
 app { 
   versionCode 1002001
  }
}

O啦,是不是很简单。
那么问题来了,这样的话就得依赖flavor。而事实上我们往往通过flavor来作为渠道的区分。
那这么一来就会引起冲突:我们既要产生多渠道,又要编译出不同的VersionCode.

排除上述方式,我的天啊。杀了我吧,搞不了!
别怕。世上无难事,只怕有心人。
在这里我首先要感谢强大的互联网,还要感谢人民,以及背后所有默默支持我的朝阳群众。
经过不屑的努力,终于被我们发现了,快到碗里来。


首先在主module下的build.gradle中定义一个方法 getAppVersionCode

def getAppVersionCode(shortVersionCode) {
  String versionName = rootProject.ext.android.versionName
  String code = ""
  if (shortVersionCode) {
    code = versionName.replaceAll("\\.", "")
  } else {
    String[] version = versionName.split("\\.");
    version.each {
      while (it.length() < 3) {
        it = "0" + it;
      }
      code += it;
    }
  }
  println code
  return Integer.valueOf(code)
}

这里我们先说一下rootProject.ext.android.versionName
通常我们为了更好的管理Android版本信息、常用依赖等,我们将这些基础配置写到一个单独的gradle文件中作为全局配置。配置ext,然后在我们的工程根目录下的build.gradle文件中做如下配置。这样就可以在全局gradle中访问ext中的信息了。

apply from: 'config.gradle'
ext {
  // 统一配置support包版本
  supportLibraryVersion  = '23.1.1'

  android = [compileSdkVersion    : 23,
             buildToolsVersion    : "23.0.3",
             minSdkVersion        : 15,
             targetSdkVersion     : 22,
             versionName          : "1.3.1",
             renderscriptTargetApi: 19,]
}

细心的同学可能会发现这里只定义了VersionName,而没有定义VersionCode。
为什么呢?山人自有妙用!

  • 好好说话
    因为我们的VersionName是不变的,而VersionCode要根据环境,然后通过调用上面定义的getAppVersionCode去赋值
    听到这里是不是有思路了。
  • 别想了,你知道在哪里赋值吗?

好了,回归正题。如果你对gradle语法还不是很清楚,可以看看这篇文章
http://blog.csdn.net/yanbober/article/details/49314255

getAppVersionCode方法就是用来根据VersionName生成versionCode值,这里接收 一个参数,长位还是短位的来得到我们需要的


方法定义好了,该怎么用呢?
注意了,注意了,重要的事情说三遍:
大招来了!!!
大招来了!!!
大招来了!!!

applicationVariants.all { variant ->
    variant.outputs.each { output ->
      // 获取渠道
      def flavor = variant.productFlavors[0]
      def flavorName
      if (flavor == null) {
        flavorName = "defaultName"
      } else {
        flavorName = flavor.name
      }
      def outputFile = output.outputFile
      def fileName
      if (outputFile != null && outputFile.name.endsWith('.apk')) {
        def isRelease = variant.buildType.name.contains('release');
        // 非release为3位code,用于区分fabric线上和dev
        // variant.mergedFlavor.versionCode 表示修改打包时versionCode的值
        variant.mergedFlavor.versionCode = getAppVersionCode(!isRelease)
        fileName =
            "${flavorName}-${variant.buildType.name}-v${variant.versionName}(${variant.versionCode}).apk";
        println fileName
        output.outputFile = new File(outputFile.parent, fileName)
      }
    }
  }

最关键的一步,你有看到吗。

variant.mergedFlavor.versionCode = getAppVersionCode(!isRelease)

没错,就是这里。我们在自定义编译后的apk重命名的同时,遍历我们的输出文件,读取apk,然后访问variant的buildtype判断是release,就设置versoncode为长位的,否则为短位的。

  • 啊?这么简单?
  • 没错就是这么简单。
这里我们简单的说一下 Variant
  • Variant就是Build Variant,中文含义 “构建变种版本”
  • Build Variant = Build Type + Product Flavor
    构建变种版本= 构建类型+定制产品
  • 我们可以通过Variant访问到productFlavors,buildType,从而可以更改我们想要更改的任意两者的相关配置
### 回答1: build.gradle 是安卓项目的一个重要文件,它包含了项目的配置信息和依赖库的引用等。下面是一个示例的 build.gradle 文件: ``` // Top-level build file where you can add configuration options common to all sub-projects/modules. buildscript { repositories { google() jcenter() } dependencies { classpath 'com.android.tools.build:gradle:4.2.0' } } allprojects { repositories { google() jcenter() } } task clean(type: Delete) { delete rootProject.buildDir } ``` 其buildscript 区块的 repositories 和 dependencies 分别指定了项目构建所需要的仓库和依赖库。allprojects 区块的 repositories 则是指定了所有子模块使用的仓库。最后,clean 任务是用来清理构建产生的临时文件。 除此之外,还可以在 build.gradle 配置应用程序的版本号、签名信息、编译选项等等。 ### 回答2: build.gradle是一个在安卓开发非常重要的文件,它用于配置和管理项目的构建过程。build.gradle文件是一个Groovy脚本,用于定义项目的各种设置和依赖关系。 首先,在build.gradle文件可以指定项目的构建工具版本编译版本。例如,可以指定使用的Android Gradle插件版本以及目标SDK版本,以确保项目能够在特定的Android设备上运行。 其次,build.gradle文件可以定义项目的依赖关系。这意味着我们可以指定项目所需的库和框架,以及它们的版本。在这个文件,我们可以通过添加依赖关系来引入第三方库,从而扩展和增强我们的应用程序功能。 除了依赖关系,build.gradle文件还可以定义构建过程所需的插件和任务。通过指定构建脚本的插件和任务,我们可以自定义和扩展构建过程,以满足项目的特定需求。 此外,build.gradle文件也包含着一些其他的配置项,如签名配置、构建类型配置等。通过对这些配置项进行设置,我们可以控制应用程序的行为和性能。 总而言之,build.gradle文件是一个配置和管理安卓项目构建过程的关键文件。通过对其的设置和依赖进行定义,我们可以定制项目的构建过程,满足特定的需求和要求。 ### 回答3: build.gradleAndroid开发的一个重要文件,它用于配置和管理项目的构建过程。它是一个Groovy脚本,位于项目的根目录下的app文件夹build.gradle文件包含了一些重要的配置项和依赖项。 首先,在build.gradle文件,我们可以设置Android项目的基本信息,如应用的包名、版本号版本名等。这些信息在构建过程会被使用,例如在生成APK文件时会用到。 其次,build.gradle文件包含了项目的依赖项配置。我们可以通过添加依赖来引入其他库和框架,以提供更丰富的功能和更好的开发体验。依赖项可以来自Maven仓库或本地文件系统。 此外,build.gradle文件还包含了构建工具的配置。例如,我们可以指定使用的编译版本构建工具版本、目标SDK版本等。这些配置项可以影响项目的编译结果和性能。 除了上述基本配置项之外,build.gradle文件还支持自定义任务的定义和配置。我们可以通过编写脚本来执行自定义的构建任务,从而满足项目的特殊需求。 需要注意的是,build.gradle文件可以包含多个模块的配置。对于大型项目,我们可以将应用程序拆分为不同的模块,每个模块都有自己的build.gradle文件。这样可以提高项目的可维护性和扩展性。 总之,build.gradleAndroid开发的一个关键文件,它用于配置和管理项目的构建过程。通过对build.gradle文件的合理配置,我们可以实现更高效、更稳定的Android应用开发
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值