在移动应用开发和运营的过程中,版本管理是一个老生常谈的基础问题,一些版本的基本概念也常常会困扰我们的研发和运营人员。同时,手动管理软件版本,也常常会因为不小心导致后续的发布和更新问题。

这里,我准备了一些 iOS 和 Android 版本的基础知识,以及如何在应用中获取版本信息和如何使用 Xcode 和 Android Studio 自动管理版本号。

版本号解释

无论是 iOS 还是 Android 都定义了两个版本属性:

  • iOS
    在 Info.plist 中定义

    • CFBundleShortVersionString
      在Xcode中解释为 Version ,这个就是我们常说的版本号,一般用户可见,通常由 <主版本号>.<次版本号>.<维护号> 三部分组成,主要用来识别不同时期不同功能的产品。

    • CFBundleVersion
      在Xcode中解释为 Build ,一般用于应用市场和程序内部识别版本,作为更新判断的依据,通常是一个递增的 INT 类型。

      这两个值可以在 Xcode 的项目信息里面进行管理,

      img_01.png

      当然,你也可以直接编辑 Info.plist

  • Android
    在 AndroidManifest.xml 中定义

    • android:versionName
      对应 iOS 中的 CFBundleShortVersionString 版本号,用作产品管理。

    • android:versionCode
      对应 iOS 中的 CFBundleVersion 编译号,作为内部识别。

      在使用 Eclipse 开发时,可以通过直接编辑 AndroidManifest.xml 文件修改,在使用 Android Studio 时,这些信息由 Gradle 脚本管理,找到并打开项目目录下 app 目录内的 build.gradle 文件,版本信息在 defaultConfig 段,

      img_02.png

程序内获取版本信息

一般在应用的关于页面,我们都会显示应用的版本,方便客服定位问题,在应用检查更新时,也常常需要用到版本信息。其实,无论是在 iOS 上,还是 Android 平台,获取笨笨信息都比较简便:

  • iOS
    索引 Bundle 信息中的相关字段:

    1. NSBundle *bundle = [NSBundle mainBundle];
    2. NSString *name = [[bundle localizedInfoDictionary] objectForKey:@"CFBundleDisplayName"];
    3. NSString *version = [bundle objectForInfoDictionaryKey:@"CFBundleShortVersionString"];
    4. NSString *build = [bundle objectForInfoDictionaryKey:@"CFBundleVersion"];
    5. NSString *fullVersion = [NSString stringWithFormat:@"version: %@ (%@)", version, build];

    上述的代码中,name 为本地化的程序名称,version 为版本号,build 为编译号,'fullVersion' 则将版本号和编译号组成一个完整的版本信息。

  • Android
    获取 PackageInfo 中的相关信息:

    1. PackageInfo pi = sContext.getPackageManager().getPackageInfo(sContext.getPackageName(), 0);
    2. String versionName = pi.versionName;
    3. int versionCode = pi.versionCode;
    4. String fullVersion = String.format("version: %s (%d)", versionName, versionCode);

    同样,versionName 为版本号,versionCode 为编译号,不同的是,在 Android 中 versionCode 使用 int 类型存储。

Xcode 和 Android Studio 编译号自增

一般应用的 版本号 都会由 产品经理 或 项目经理 决定,根据产品所处的阶段和功能决定修改版本号的哪一个段。但 编译号 更多时候是根据每次发布递增,有时候还会遇到同一个版本号对应多个编译号的情况,如,紧急修复了一个关键 Bug 并更新发布一个版本,但如果每次发布都要手动去修改编译号回很繁琐,也很容易忘记和出错,而不管是 Xcode 还是 Android Studio 都没有提供自增编译号的功能,我们只有手动添加编译脚本来达到自增的目的。

  • Xcode
    添加编译过程,读取并修改 Info.plist 中的版本信息:

    1. 打开工程,选择编译目标,点击 Build Phases 选项卡。

      img_03.png
    2. 点击左上角的 + 添加,选择 New Run Script Phase 添加一个编译脚本。

      img_04.png

      img_05.png
    3. 在脚本编辑其中输入下面的脚本:

      1. #!/bin/bash
      2. buildNumber=$(/usr/libexec/PlistBuddy -c "Print CFBundleVersion" "$INFOPLIST_FILE")
      3. buildNumber=$(($buildNumber + 1))
      4. /usr/libexec/PlistBuddy -c "Set :CFBundleVersion $buildNumber" "$INFOPLIST_FILE"

      上述脚本使用 PlistBuddy工具,读取 Info.plist 中的 CFBundleVersion 值,+1 后写回 Info.plist 中。

    4. 我们需要让该段代码在应用打包前执行,以便将修改应用到App包内,因此我们将该编译过程重命名为 Increase Version Code 并移动到 Copy Bundle Resources之前。

      img_06.png
    5. 这样,当我们每次编译该 Target 时 ,就会执行改脚本将 CFBundleVersion 值增加1,但没次调试运行时,该值都会加1,这并不是我们想要的,我们只需要在每次打包发布时增加1就行,因此,我们将 Run script only wehn installing 前的勾打上,这样就只会在打包应用时执行改脚本。

      img_07.png
  • Android Studio
    Android Studio 的 versionCode 自增原理和 Xcode 类似,不同的是我们不能让编译脚本修改自身,而是通过一个额外的 Java Properties 来文件存储版本信息:

    1. 打开工程,选择 Module: app 的编译脚本 build.gradle

      img_08.png
    2. 找到 defaultConfig 段,替换为下面的脚本:

      1. def versionPropsFile = file('version.properties')
      2. if (versionPropsFile.canRead()) {
      3. Properties versionProps = new Properties()
      4. versionProps.load(new FileInputStream(versionPropsFile))


      5. def verCode = versionProps['VERSION_CODE'].toInteger()
      6. versionProps['VERSION_CODE'] = (++verCode).toString()
      7. versionProps.store(versionPropsFile.newWriter(), null)


      8. defaultConfig {
      9. applicationId "com.example.versionexample"
      10. minSdkVersion 19
      11. targetSdkVersion 23
      12. versionCode verCode

      13. versionName "1.0.1"
      14. }
      15. } else {
      16. throw new GradleException("Could not read version.properties!")
      17. }

      这段脚本会打开 app 目录下的 version.properties 配置文件,读取 VERSION_CODE 字段并增加1后写回,同时将值赋给 defaultConfig 中的 versionCode 。

    3. 此时,点击 Sync 会报错 Could not read version.properties!,这是我们刚刚在脚本中抛出的,原因是找不到 version.properties 文件,我们需要新建一个。打开命令行定为到项目目录下:

      1. $ cd app/
      2. $ echo "VERSION_CODE=1" > version.properties

      再次点击 Sync 就不会报错了。查看 version.properties 文件,

      1. $ cat version.properties
      2. #Mon Nov 02 15:18:49 CST 2015
      3. VERSION_CODE=3

      发现 VERSION_CODE 已经增加到 3 了,说明该脚本被执行了两次,但这并不符合我们的预期。

    4. 同 Xcode 中一样,我们期望仅仅在应用打包时将 versionCode 增加 1 。因此我们需要获取编译参数,仅当 release 时才将 versionCode 增加 1 。修改后的脚本如下:

      1. def versionPropsFile = file('version.properties')
      2. if (versionPropsFile.canRead()) {
      3. Properties versionProps = new Properties()
      4. versionProps.load(new FileInputStream(versionPropsFile))


      5. def verCode = versionProps['VERSION_CODE'].toInteger()
      6. def runTasks = gradle.startParameter.taskNames
      7. if (':app:assembleRelease' in runTasks) {
      8. versionProps['VERSION_CODE'] = (++verCode).toString()
      9. versionProps.store(versionPropsFile.newWriter(), null)
      10. }
      11. defaultConfig {
      12. applicationId "com.example.versionexample"
      13. minSdkVersion 19
      14. targetSdkVersion 23
      15. versionCode verCode

      16. versionName "1.0.1"
      17. }
      18. } else {
      19. throw new GradleException("Could not read version.properties!")
      20. }

      这里获取当前task的名称,仅当task包含 :app:assembleRelease 时才会将 versionCode 加 1 ,否则直接使用读取到的值。

小结

版本管理是产品和项目管理中非常重要的一环,但在开发初期也常常容易被忽略,等到遇到问题时候才感到头痛。其实,产品经理和开发人员在产品的立项阶段就应该对产品版本有一个一致的理解,我个人通常在项目框架建立之初就将这些规则脚本添加到项目文件中,以期降低后续版本维护的成本。

使用 Xcode 和 Android Studio 管理 iOS 和 Android 项目版本的更多相关文章

  1. 【转】如何使用Android Studio把自己的Android library分发到jCenter和Maven Central

    转自:http://www.devtf.cn/?p=760&utm_source=tuicool 如何使用Android Studio把自己的Android library分发到jCenter ...

  2. Android Studio之同一窗口打开项目

    Android Studio默认新打开的项目都是重新打开一个窗口,和原项目窗口同时存在,如果打开多个项目,则有很多窗口同时打开,怎么根据需要决定自己以何种方式打开呢? 1.设置打开新项目的方式 第一项 ...

  3. Android Studio安装以及Fetching android sdk component information超时的解决方案

    转载:http://www.cnblogs.com/sonyi/p/4154797.html 在经过两年的开发之本后,Google 公司终于发布了 Android Studio 1.0,喜欢折腾的童鞋 ...

  4. 利用Android Studio、MAT对Android进行内存泄漏检测

    利用Android Studio.MAT对Android进行内存泄漏检测 Android开发中难免会遇到各种内存泄漏,如果不及时发现处理,会导致出现内存越用越大,可能会因为内存泄漏导致出现各种奇怪的c ...

  5. (mac)Android Studio安装以及Fetching android sdk component information超时的解决方案

    解决Mac下面Fetching android sdk component information加载过久问题, 关于windows中可以参考前面一篇文章 关于安装和下载可以百度一下地址.安装完成后, ...

  6. (window)Android Studio安装以及Fetching android sdk component information超时的解决方案

    转自:http://www.cnblogs.com/sonyi/p/4154797.html 在经过两年的开发之本后,Google 公司终于发布了 Android Studio 1.0,喜欢折腾的童鞋 ...

  7. 解决Android Studio启动速度慢的问题。避免每次启动Android Studio都要fetching Android sdk compoment information。

    Android Studio每次启动都要去fetching sdk,由于Android sdk 官网在大陆连不上,所以每次启动时界面都会停在那里很久. 解决办法就是设置取消每次fetching sdk ...

  8. Android Studio 快速实现上传项目到Github(详细步骤)

    前言: 本文主要讲解如何将Android Studio项目上传至GitHub,在此之前,先介绍几个概念. Android Studio:是谷歌推出一个Android集成开发工具,基于IntelliJ ...

  9. Android studio 使用心得(六)—android studio 如何加载.so文件

    之前一直没怎么注意,以为.so文件android为像eclipse一样直接加载,但是直到昨天我在android studio上调试公司项目推送消息的时候,才发现,.so文件原来没有加载成功. 可能之前 ...

随机推荐

  1. 关于Keil的安装与注册

    由于前一段时间一直在做关于stm32f407的相关内容,于是安装的Keil是MDK5,最近一阵子想再看看51单片机以前没有做过的内容,就要再安装一个Keil C51,结果就不可避免的遇到了两个软件必须 ...

  2. windows“画图”工具用法

    图片编辑工具不少,photoshop功能强大,但是并不是那么容易掌握.而且,倘若一个简单的图片处理问题就使用这么强大的工具,有点“杀鸡焉用牛刀”的意思,要知道只打开photoshop就需要一段等待时间 ...

  3. 《深入剖析Tomcat》阅读(一)

    第一章 一个简单的Web服务器 该应用程序仅接受位于指定目录的静态资源的请求,如HTML文件和图像文件.它也可以将传入的HTTP请求字节流显示到控制台上.但是,它并不发送任何头信息到浏览器,如日期或者 ...

  4. 利用servlet做转发,实现js跨域解决同源问题

    做前端开发,避免不了跨域这个问题,跨域具体什么概念,不赘述,博客里太多.简单说下,我们用js发请求,不管post还是get,如果发请求的对象和当前web页面不在同一域名下,浏览器的同源策略会限制发请求 ...

  5. [BZOJ 1014] [JSOI2008] 火星人prefix 【Splay + Hash】

    题目链接:BZOJ - 1014 题目分析 求两个串的 LCP ,一种常见的方法就是 二分+Hash,对于一个二分的长度 l,如果两个串的长度为 l 的前缀的Hash相等,就认为他们相等. 这里有修改 ...

  6. codeforces C. Jzzhu and Chocolate

    http://codeforces.com/contest/450/problem/C 题意:一个n×m的矩形,然后可以通过横着切竖着切,求切完k次之后最小矩形面积的最大值. 思路:设k1为横着切的次 ...

  7. Weblogic8.1 的性能优化

    注:在下面做的介绍都是以Weblogic8.1为例的,其它版本的Weblogic可能会有些许不同. 1) 设置JAVA参数: a) 编辑Weblogic Server启动脚本文件: BEA_HOMEu ...

  8. 中文乱码 $dbh->do("SET NAMES utf8");

    use DBI; my $dbUser='DEVOPS'; my $user="root"; my $passwd="kjk123123"; my @arr2= ...

  9. [LeetCode#202] Roman to Integer

    Problem: Write an algorithm to determine if a number is "happy". A happy number is a numbe ...

  10. vijosP1779国王游戏

    题目:https://vijos.org/p/1779 题解:忽然想起来我好像还没写过高精度除以单精度,于是拿这题练练手...没想到1A了... 代码: #include<cstdio> ...