使用 Xcode 和 Android Studio 管理 iOS 和 Android 项目版本

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

这里,我准备了一些 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 信息中的相关字段:

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

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

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

    PackageInfo pi = sContext.getPackageManager().getPackageInfo(sContext.getPackageName(), 0);
    String versionName = pi.versionName;
    int versionCode = pi.versionCode;
    
    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. 在脚本编辑其中输入下面的脚本:

      #!/bin/bash
      buildNumber=$(/usr/libexec/PlistBuddy -c "Print CFBundleVersion" "$INFOPLIST_FILE")
      buildNumber=$(($buildNumber + 1))
      /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 段,替换为下面的脚本:

      def versionPropsFile = file('version.properties')
      if (versionPropsFile.canRead()) {
          Properties versionProps = new Properties()
          versionProps.load(new FileInputStream(versionPropsFile))
      
          def verCode = versionProps['VERSION_CODE'].toInteger()
          versionProps['VERSION_CODE'] = (++verCode).toString()
          versionProps.store(versionPropsFile.newWriter(), null)
      
          defaultConfig {
              applicationId "com.example.versionexample"
              minSdkVersion 19
              targetSdkVersion 23
              versionCode verCode
              versionName "1.0.1"
          }
      } else {
          throw new GradleException("Could not read version.properties!")
      }
      

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

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

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

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

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

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

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

      def versionPropsFile = file('version.properties')
      if (versionPropsFile.canRead()) {
         Properties versionProps = new Properties()
         versionProps.load(new FileInputStream(versionPropsFile))
      
         def verCode = versionProps['VERSION_CODE'].toInteger()
      
         def runTasks = gradle.startParameter.taskNames
         if (':app:assembleRelease' in runTasks) {
            versionProps['VERSION_CODE'] = (++verCode).toString()
            versionProps.store(versionPropsFile.newWriter(), null)
         }
      
          defaultConfig {
              applicationId "com.example.versionexample"
              minSdkVersion 19
              targetSdkVersion 23
              versionCode verCode
              versionName "1.0.1"
          }
      } else {
          throw new GradleException("Could not read version.properties!")
      }
      

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

小结

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

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,734评论 25 707
  • 近期有一个工作任务:按照某个规则,给Android应用设置一个在编译时自动生成的versionCode与versi...
    匿蟒阅读 20,526评论 11 28
  • Rstudio-server连接出错,如题。 重新编译,加入--enable-R-shlib选项 R升级到3.2后...
    gada阅读 1,525评论 0 1
  • 书柜里有好多好书,一直都没有时间去好好的看。最近,总是没来由的随手拿起一本,然后被它吸引。 《黑暗,也是一种...
    Gracee224阅读 2,467评论 0 0
  • 文/午言 提起母亲两字 儿子泪如雨下 愧疚感负身不能自已 只能悄悄的在旁边擦干眼泪...
    許勇阅读 481评论 4 2