这是Cyril Mottier最近更新的一篇文章,原谅地址在这里:Android开发方法学

这篇文章是他介绍自己所在项目小组(Capitaine Train Android Team)设计、开发时的一些流程,其实并没有什么有关Android代码方面的什么干货,但我觉得看一下仍能给我们一些APK生产过程中一些启示,比如APK的版本问题等。

下面是博客的原文,有兴趣的可以看一下:

软件开发方法学

我最近常被问到Capitaine Train的Android团队是如何工作的:“你们多久发布一次应用新版本?”,“你们的版本管理策略是怎么样的?”,“你们公测测试构建(test builds)吗?”等等。让我们先说明白:我不是一个流程学或方法学怪胎,且在大多数时间选择对此缄默。我宁愿把自己看作产品男。然而,不管产品有多大,软件开发流程学都是产品成功的一部分。本篇尝试分享和讨论Capitaine Train中的Android团队所使用的“工作方法”。

在深入了解这篇文章之前,让我们快速放弃一些不恰当的期待。的确,我觉得提及策略是肯定的,以下描述的进程学和其它的方法学远非完美。就像在UX和开发中没有完美的答案一样,处理大型项目也没有完美的方法。换句话说,本文无意迫使你转向新的方法学,反而,它应该被当作简单的回馈来阅读,当然,是有关团队如何管理像Capitaine Train Android应用一样的应用。

Capitaine Train对于Android 环境条件

方法学在脱离了使用的环境条件时是没有任何意义的。因此我想念,简要介绍一下Capitaine Train中的Android团队是必要的。这个团队诞生于2013年3月份,那时我加入Capitaine Train来带领Android应用。习惯极端复杂的欧洲培训生态系统花费了我相当一段时间。我们快速地决定使该团队发展壮大。2013年11月份的时候Mathieu Calba加入,之后在2014年10月份Flavien Laurent加入。如果你真正地擅长算术的话,你本已注意到Capitaine Train的Android团队只是个3人小组!为了更好地理解我们的流程学,要记住的重要一点是,成立这样一个相对微小的团队。

我十分自豪成为Capitaine Train的一部分,但更自豪的是,我新手建立了Capitaine Train中的Android团队。Mathiue和Flavien每一天都会使我感到惊喜,与之共事问题一件人生快事。所有的成员都绝顶聪明,且极度专注于产品,毫不犹豫地为自己的信仰而战,且对优良的UI/UX(并不真的跟开发人员理解地相同)有清晰的理解。换句话说,在涉及到重新设计新特性时,团队的每一个成员都极端自立。他们的大多数时间都花费在Android相关的特性上。似乎我们全都工作在Android framework上面,但有时也花费些时间在别的平台上(例如Ruby on Rails)来实现特别是跟Android有关的特性(Google Now就是一个极好的例子)。

从产品的视角来看,Capitaine Train中的Android可以总结为两种不同的应用。最重要的部分,即手持设备,适配了平台和手机。最小也是最新的部分,则以可穿戴设备为目标。可支持的最小版本是Android 4.0,且应用绑定了四种不同的语言:英语,法语,德语和意大利语。

Capitaine Train的目标消费者是全球范围的。然而,我们的产品很清晰地专注于欧洲火车。由此,我们的大多数观众以欧洲为基础,且生活在2~3个不同的时区。必须处理这样一个相对较小的时区范围是相当有益的,尤其是同步的声明新的东西作为沟通(在某各方式下,一天中的给定时刻对于所有的用户而言是近乎相同的)的工具时。

对于下载次数,我们通常不太关注。然后,在写这篇文章的时候,Google Play Store显著地表明这个应用已经下载了5~10万次。

实现流程

讨论Capitaine Train如何处理代码也许是整篇文件的主题。没有深入于设计,我将只介绍一下我们开发和测试技的流程。

谁开发,谁复查

完整的Android代码基础是通过Git管理的。我觉得本文中呈现了Git这个词是不必要的,尽管它作为最佳源码控制管理系统之一已经为众人所知。我们在Capitaine Train中广泛地使用Git,我们所有的复查技术都是基于这件令人拍案叫绝的开发工具。

在Git之上,我们也使用git flow模型(http://nvie.com/posts/a-successful-git-branching-model/)

这个模型确保了连贯可理解的commits树。简单地讲,就是开发在dev分支上完成。当队员需要开发新的特性时,一个新的“特性分支”就会从dev中创建出来。这个分支的创建取决于负责这个特性实现的队员。在这个“特性分支”完全完成之后,会重新合并到dev分支上面。发布应用和通过使用--on--off选项将dev合并到master是同步的。这个选项确保了合并操作总是使用commit来表示。最终这次commit标上了应用的版本码。在其它方面,master应该只包含涉及应用公共版本的commits。

在合并之前,代码问题应该被团队中至少两名成员阅读和检查。

因为我们的Android团队十分小,所有的特性总是由单个成员来管理的,如Bob。Bob对特性的开发完全负责:从设计到发布。一旦特性被认为成熟了,打磨好了,它将作为一个“合并请求”提交给Android团队的另外一个成员(Alice)。代码复查的完成得感谢称为GitLab的工具,它可以看作GitHub的副本。Alice负责代码复查。复查的反馈是错综复杂的。举例来说,这样的复查很普遍:“你应该使用这个方法”“我用了不同的文本颜色和大小”“如果APK大于5MB,我是不会合并的”。相较于一个简单的“No!”,拥有可选说明集合的复查通常更加合理。

Capitaine Train Android团队的特色之一是每个成员都会负责QA。确实是,公司里面没有QA团队。Bob和Alice必须确保没有回退且代码运行完美。它基本上意味着代码复查不只包含详读代码。Alice也必须测试在所有可能情况下的特性实现。

通常而言,一旦Bob和Alice对特性说了OK(代码,设计,引入的改变,API等),代码复查流程就结束了。此后,在特性分支合并到dev之前,Capitaine train Android应用的代码问题会被团队中的至少两名成员阅读和检查。这事实上适用于Capitaine Train中的所有项目。大的公司也会依靠类似的流程并且要求至少两个来自复查人员中的”+1”。很显然我们是如此小的团队,力不足以这么做啊。

打包构建

完整的项目构建在Gradle之上。新的Android基于Gradle的构建系统的主要优势事实上依赖于这个事实:它适用于开发和打包的目的。在开发一个特性的时候,我们全部都用Android Studio(也使用了Gradle);而当涉及打包的时候,我们将使用命令行。构建打包的完成得亏于我们持续地完善基于Jenkins的环境。Jenkins当前管理两个不同的Android应用:

  • Android-dev,构建处于当前dev分支上面的项目
  • Android-master,相应于master分支

当主远程repository(origin)中新的commit推送到时,这些项目的新的构建过程将会触发。主要的不同是每个工程指向不同的分支,并且拥有不同的编译选项。的确,与android-dev相反,android-master的优化和混淆得亏于Proguard。

由于我们必须创建大量的截屏(3种外形尺寸,4种语言,6种截屏:3x4x6=72个截屏),最近添加了新的自制工具到打包流程中:自动截屏(感谢Flavien)。新的工具负责按照所有必要的配置来截取屏幕,并通过统一状态栏的方式来清理截屏。在写作本文的时候,这个工具还没有集成到我们的Jenkins构建流中但这铁定了是将来要完成的事。

当谈到在Google Play Store上面发布时,一切都以手动完成。显然,新的Google Play发布API可以使用,但目前为止我们倾向于保持对版本的控制。由于我们“漫长的”发布生命周期,我确信这在我们当前处理发布的方式上面不是什么问题。

应用版本管理

管理Android应用的版本是很必要的流程。的确,Google Play Store为了侦别应用新版本使用了应用版本号。来自Google Play Store的惟一需求是确保应用版本号单调递增。

并没有尝试分辨每一个版本是主要版本、次要版本还是补丁,每一次包含至少一个新的用户可见特性的发布都被认为是主要版本。

Capitainne Train的Android应用没有使用传统的主要版本-次要版本-补丁的版本管理方式(详查http://semver.org/)。的确,由于想要尽可能少的麻烦,我们想出了一个更简单的基于两个版本数字---主要和次要---的版本管理模型。应用码通过基于这些数字利用下面的公式计算得来:

隐藏在这个版本管理策略之后的主要想法是在发布流程中没有或者有但尽可能少的摩擦。并没有抓破脑袋尝试分辨每一个版本是主要版本、次要版本还是补丁,每一次包含至少一个新的用户可见特性的发布都被认为是主要的。Bug修复和补丁问题认为是次要版本。当如下描述的发布时间表联系起来时,这种方式工作起来尤其好。

从外部用户的视角来看,只有主要版本是重要的。应用的版本名总是主要版本而不是次要版本。隐藏在这种命名策略之后的主要原因是:当次要版本并不包含任何用户可见的特性时,他们理应对用户完全透明。只使用主要版本名号使得记忆更加容易和更多的可识别度。如果你果真要了解应用的精确版本号,你可以打开Capitaine Train Android应用的“设置”屏幕。它展示了版本名(当然,也展示在了系统设置应用中)+版本号。

开发是有趣的,但你本可以使之更加有趣。我们所有的公共构建事实上内部命名的(就像常规的Android发布)。作为一个超级Stargate SG-1粉丝,我根据其中的一些角色命名每一个主要发布:ANUBIS(101),BRATAC(201),CARTER(301)等。次要发布是使用了后缀_MR<x>的名字,其中<x>是次要版本号减一:FRAISER_MR1(602),GEORGES_MR1(702).

Android Wear应用版本也同样遵循相同的版本管理模式,并依据Pixar电影角色名来命名:ANDY,BUZZ,COLETTE等。尽管从发布的视角来看,可穿戴设备应用和手持设备应用绑定在一起(可穿戴设备APK打包在手持设备APK里面并在Play Store上同时发布),我们决定使用明晰的版本管理方式。

内部玩乐显然不是维护这么一个版本号列表的唯一目的。它协助我们依据应用的当前版本轻易地改变执行行为。例如,在给定版本的应用上在硬盘上存储一些东西时遇到了严重的问题,你可能想要检查一下应用的版本是否大于存储在硬盘上的数据来获知是否是时候来更新数据到新的格式。

完整的发布流程

Capitaine Train用户可能已经注意到,这个团队非常专注于这个产品。我们并没有发布新特性,直到这些新特性完成准备好了生产。我们高质量的标准防止我们公开发布非打磨精良的特性。这是本产品没有严格的截止日期的原因之一:

没有坚持严格的截止日期,这个Android应用遵循发布火车软件发布时间表。

并没有坚持严格的截止日期,这个Capitaine Train Android应用遵循发布火车软件(http://en.wikipedia.org/wiki/Software_release_train)发布时间表。这个发布培训是基于时间的发布时间表。它既不等待特性,也不等待bug修复,但却尽可能纯粹地基于时间。简单来讲,这个应用的每一个新版本都可能被认作准时来去的火车。如果特性在按计划到来的时间准备完毕,它将跳入这辆发布火车。如果没有,这个特性将必须等待下一辆发布火车。发布火车迫使规律地引进特性,给予可预测性,允许更多常规发布。And Yes!方法学命名和公司的名字之间的相似性纯属巧合:-)

发布火车仅仅适用于Capitaine Train Android应用的主要发布。次要发布在完全不同的时间线上完成。因为次要版本通常热修复阻塞和崩溃问题,他们尽可能快地发布,而不考虑发布火车时间表。当然,这仅发生在该应用的公测阶段,这将再接下来的文章中讨论。

发布生命周期

Capitaine Train Android应用新主要版本的发布遵循递归模式。这种模式每6周重复自身一次。为什么特性是6呢?坦白地讲,这个计算后面没有复杂的数学。这仅仅是经验,源于我作为Android应用设计者和开发者的经历。这里是解释:

  • 默认情况下,Android自动更新应用。更新的时候,通知展示在通知屉中。过于频繁地发布应用可能惹恼用户,以致他认为你的应用是垃圾邮件一类的东西。
  • 过于频繁地发布应用新版本同义于小的特性更新。这使得发布减少了吸引力,更为重要的,减少了市场号召力。
  • 在另一方面,使用更长的时间周期发布也可能适得其反。用户将会完全忘记你的应用。此外,在每一次应用发布中,也增加了产生严重问题的潜在可能性。
  • 考虑到我们的开发/产品团队和产品进化的步伐,我们确信每6周就能够引进新的用户可见的特性。历史向我们展示自从版本1之后,我们设法为每一次发布坚持这个时间(但却不是版本1.0)

坦白地讲,我不认为存在完美的发布生命周期长度。在Capitaine Train,这个6周模式工作地尤其好,因为它来自用户和我们自身期待的折衷。

以上表格描述了我们的发布生命周期。就像早前预料的一样,每一个版本v(n) 都会准备7周。因为与之后的生命周期交叠一起,新版本6周发布一次。发布周期内的计划是非常灵活的,并由工程人员决定。然而,它通常缩减为:

  • 第一周与产品经理会面,讨论并评估特性优先级(例如在需求阶段、实现阶段等)。特性排序阶段事实上通常提前完成,但是经常得在第一周内。
  • 第一周至第四周大多数时间持续开发新特性。
  • 第五周完成半数的新特性开发和Bug修复。因为我们想要确保复查人员拥有至少一周时间来复查代码,在第五周末的时候代码是被认为是“特性冻结”的。通常而言,留给代码复查的时间是灵活的,并取决于特性的综合复杂度。
  • 在第六周,工程师集中精力于Bug修复、代码复查和完整性测试。本周的主要目的是在本周末尾的时候打磨并准备产品的公测。

由于在版本v(n) 的第七周应当没有安排什么事(至少从开发的视角来看,本周不止是公测周),本周也是v(n+1) 版本的第一周。如果在公测阶段没有大的事故发生,那么公测频道发布的构建将会在本周末提升为生产。万一在公测阶段有大的bug出现,那么将由引进这个bug的成员负责修复并在本周末处理接下来的构建和发布流程。

发布日是这样的……

另外有趣的一点是Capitaine Train android应用问题发布在一周的相同一天:周二。更精确一点的话,是周二早晨。和一周的其它时间相比,周二有几个优势。这样发布的大多数原因是对于大多数软件项目是通用的,但有一些别的是显示十分个性:

众所周知,周二早晨通常被认为是发布新东西的最佳时刻。这里面大部分的原因是周二是一周中最忙碌的一天。周二发布应用是帮助你们的市场开拓团队最好的方式,并且能够确保对用户影响最大化。

周二是一周中第二天(这不会令人感到惊讶是吧?呃,它其实依赖于你认为周首日的方式。在我看来,一周中的第二天是周一。)。万一公开发布转变成了bug和崩溃的戏剧表演,开发/支持团队则在周末之前有至少3天半的时间来修复bug和崩溃问题。毫无疑问,工作日要比非工作日具有更清晰的意识。

通常,周二对于Android团队而言是“镇静”的一天。我的同事Mathieu和我都住在里昂(跟巴黎500公里),一周里面,在巴黎的办公室工作3天,而在家工作2天。因为我们通常在周四和周五远程工作,所以我们倾向于将会议和讨论累积到周一以清醒我们的意识并为之后顺利地发布做准备。

公测频道,崩溃报告和舞台首演在同一条船上

我先前提到过Capitaine Train Android发布生命周期的公测阶段。在Capitaine Train,公测是通过Google Play Store的公测频道完成的。公测是私密的,人们可以请求加入。将牵涉到添加新的公测人员到我们的公测池时,我们是非常挑剔的。的确,公测人员必须既极度活跃(用户至少每周一次火车旅行)又值得依赖(我们不想让他与别人交流有关即将发布的特性的事)。公测人员有整整一周时间来测试和报告错误。

由于Google Play Store上面的崩溃报告要求用户同意发送崩溃信息,我们添加了额外的崩溃报告器:Crashlytics。当用户不想报告崩溃时,这将极端地重要。例如,对于当前的产品版本,Google Play Store和Crashlytics错误报告上的不同有有多达25倍的差异。Crashlytics帮助我们接到重要的FC(Fatal Crash)。我们也能够通过发生的次数和设备类型对崩溃进行优先级的确定。

Google Play Store提供了一个称作“舞台首展”的很好的功能。舞台首展使得你的应用的新版本只对你整个用户基础上的一个子集可见。例如,它允许你发布只对10%用户的新的构建。这对于测试新特性或者潜在在缩减服务器负荷有显著的好处。在第一个Android版本的Capitaine Train期间,我们一直跟这个特性相处。因为我们对自己的发布生命周期十分自信,我们现在很少使用它。因此,我们95%的发布是通过单次全部发布完成的。

总结

我想我已经详细解释了Capitaine Train Android团队是如何运转的。我尝试尽可能准确的描述,但可能难免忘记一些重要的点。要毫不犹豫地留下评论,我将尽量回答有关遗失信息的问题。再一次提醒,不要忘记这篇文章描述了完美应用于Capitaine Train Android团队的方法学。总是要记住,你我所有的环境条件都是不同的。在改变和适配你在自己项目上的工作方式之前,请考虑一下子你工作的环境条件。

Android开发方法学的更多相关文章

  1. 【原】Android热更新开源项目Tinker源码解析系列之三:so热更新

    本系列将从以下三个方面对Tinker进行源码解析: Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Android热更新开源项目Tinker源码解析系列之二:资源文件热更新 A ...

  2. 配置android sdk 环境

    1:下载adnroid sdk安装包 官方下载地址无法打开,没有vpn,使用下面这个地址下载,地址:http://www.android-studio.org/

  3. Android SwipeRefreshLayout 下拉刷新——Hi_博客 Android App 开发笔记

    以前写下拉刷新 感觉好费劲,要判断ListView是否滚到顶部,还要加载头布局,还要控制 头布局的状态,等等一大堆.感觉麻烦死了.今天学习了SwipeRefreshLayout 的用法,来分享一下,有 ...

  4. Android Studio配置 AndroidAnnotations——Hi_博客 Android App 开发笔记

    以前用Eclicps 用习惯了现在 想学学 用Android Studio 两天的钻研终于 在我电脑上装了一个Android Studio 并完成了AndroidAnnotations 的配置. An ...

  5. Android请求网络共通类——Hi_博客 Android App 开发笔记

    今天 ,来分享一下 ,一个博客App的开发过程,以前也没开发过这种类型App 的经验,求大神们轻点喷. 首先我们要创建一个Andriod 项目 因为要从网络请求数据所以我们先来一个请求网络的共通类. ...

  6. 【原】Android热更新开源项目Tinker源码解析系列之一:Dex热更新

    [原]Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Tinker是微信的第一个开源项目,主要用于安卓应用bug的热修复和功能的迭代. Tinker github地址:http ...

  7. 【原】Android热更新开源项目Tinker源码解析系列之二:资源文件热更新

    上一篇文章介绍了Dex文件的热更新流程,本文将会分析Tinker中对资源文件的热更新流程. 同Dex,资源文件的热更新同样包括三个部分:资源补丁生成,资源补丁合成及资源补丁加载. 本系列将从以下三个方 ...

  8. Android Studio 多个编译环境配置 多渠道打包 APK输出配置

    看完这篇你学到什么: 熟悉gradle的构建配置 熟悉代码构建环境的目录结构,你知道的不仅仅是只有src/main 开发.生成环境等等环境可以任意切换打包 多渠道打包 APK输出文件配置 需求 一般我 ...

  9. JS调用Android、Ios原生控件

    在上一篇博客中已经和大家聊了,关于JS与Android.Ios原生控件之间相互通信的详细代码实现,今天我们一起聊一下JS调用Android.Ios通信的相同点和不同点,以便帮助我们在进行混合式开发时, ...

随机推荐

  1. Codeforces 420 B. Online Meeting

    B. Online Meeting time limit per test 1 second memory limit per test 256 megabytes input standard in ...

  2. windows平台搭建lighttpd+php+sqlite

    (一)php 1. 下载及安装 http://www.appservnetwork.com/ 从上面的网址下载appserv-win32-2.5.10并安装,在安装的时候,仅仅选择安装php. 由于, ...

  3. android使用ffmpeg

    cygwin上文编译文章. 在ffmpeg/arm添加的文件夹Android.mk 的主要目的是为了宣布动态库libs下一个 LOCAL_PATH:= $(call my-dir) include $ ...

  4. 实现 mouse-drag 的图标拖动

    <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/stri ...

  5. nginx.conf 完整的集群配置

    ###############################nginx.conf 整配置############################### #user nobody; # user 主模 ...

  6. 三个水杯 (bfs)

    给出三个水杯,大小不一,并且只有最大的水杯的水是装满的,其余两个为空杯子.三个水杯之间相互倒水,并且水杯没有标识,只能根据给出的水杯体积来计算.现在要求你写出一个程序,使其输出使初始状态到达目标状态的 ...

  7. 小工具:内存监视器(SystemMonitor)

    卸了360之后,与之捆绑的加速球也没了.加速球可以查看剩余内存量,清理残留进程,有的时候不觉得这小玩意有多大作用,卸了之后才知道后悔. 加速球的替代方案比比皆是,如Windows自带的任务管理器,窗口 ...

  8. 【软件project】生存期模型(含图)

    为了反映软件生存周期内各个工作应怎样组织,各阶段怎样衔接,须要软件开发模型给出直观图示表达.软件开发模型是软件思想的详细化,是实施在过程模块中的软件开发方法和工具. 以下来介绍开发模型的特点以及他们的 ...

  9. Rightmost Digit(快速幂)

    Description Given a positive integer N, you should output the most right digit of N^N.               ...

  10. js isArray小结

    原文:[转载]js isArray小结 在日常开发中,我们经常需要判断某个对象是否是数组类型的,在js中检测对象类型的常见的方法有几种: 1.typeof操作符.对于Function.String.N ...