iOS大型项目开发漫谈

标题有些吓人请不要害怕,不过这确实不是扫盲贴,需要一定的iOS开发基础。在我多年的码农生涯中绝大部分时间都是做的小项目,大一些的可能也就是百万行代码的样子,跟Windows系统几千万行源码比简直就是小巫见大巫。不过,一个iOS项目的源码有数百万行算蛮大了。我想说的是,人总是会成长,会担当更大的责任接受更大的挑战,终有一天组织会有重要任务交给你。不过软件开发不是一朝一夕,也不会有多么的轰轰烈烈,更多的是平淡中无数的细节处理。做大型项目未必就需要多么高深的技术,也许就是细节的科学处理与规范的管理。

首先说说编程语言的选择,Objecive-C还是Swift?我还没有在项目中使用Swift,因为我说服不了自己去用它,它的优势在哪里,你也不能强迫队友去学习Swift。当然,不用不代表不会,一入行就用Swift开发无意义产品的人没资格戴着有色眼镜鄙视不会Swift的同行。你知道Objecive-C与Swift混编有多少坑吗?你知道Swift也是跟Objecive-C共用一个Runtime环境吗?你知道使用了Swift会使得ipa包大10多M吗?Swift代表未来,Objecive-C是当下。Swift能做的Objective-C都能做,比如Closure完全就可用Blocks来代替,Tuple完全可以用结构体来代替。生产环境用Objective-C,业余观望Swift,这就是我的态度。Raywenderlich已经出了一堆Swift的教程,在前沿科技的使用上国内总是慢一个节拍,很大程度上那堵墙阻碍了国内IT从业者的发展,墙的开发者(包括我的信息安全工程老师)终有一天会受到历史的审判。

再来说说应用架构,真是成也MVC,败也MVC。如果放任团队中的小朋友按他们所理解的MVC来开发,一般情况下出现的恶果就是类之间高耦合,Controller胖得不像话简直就成了百宝箱......每一次的版本迭代都会痛苦不堪,若功能不多还能勉强维持功能多了势必崩塌,就比如世博园中国馆吧,再按比例多盖5层试试看,呵呵。到头来开发人员实在受不了就只能选择重构要么跑路,后者更现实大家都懂,尤其是业务为王的企业里,代码质量算个屁只要能work,可是好的程序员都是有尊严的,deadline或是拍脑袋的政治任务通常会让程序员疲于应付,厌倦了也就该撤了。言归正传,既然MVC显得臃肿,那就是瘦身呗。通常瘦身后的MVC在iOS界被叫成MVCS或MVVM,这2个当然不是同一个东西,项目选用MVCS还是MVVM还是得看你的业务特性。MVCS与MVVM就那么完美吗?当然不,MVCS要注意Service/Store的滥用,其中的数据是否会被不同的模块污染。MVVM用得不好也会增加项目的维护难度,我说的是View和ViewModel之间松散的绑定关系,在iOS开发中一提到MVVM大家就想到ReactiveCocoa,其实没有ReactiveCocoa也可以啊,只是ReactiveCocoa的好处多多,如代码收敛不必写得到处都是,其他好处自己去挖掘吧。

关于工程文件组织结构,很多人是Controllers/Views/Models/Vender...这样归类,其实这有个问题,比如你要找ControllerA的TableView用到的cellB类,你还要去Views里面一个个找,太痛苦了,就算search也还是苦毕竟不能所见即所得。我一般是按 Module来划分,每个Module里面有Controllers/Views/Models/Stores/API这样的分类,API是每一个接口的请求的封装,供Store调用,得到的数据解析实例化为Model再通过block传递给Controller去刷新View,很绕吗?使用RAC,Controller订阅Store里创建的RACSignal也能做到,U can make a try。还有就是Helper与Utility,与业务无关,具有对象性质,提供支持功能的代码放到Helper,比如创建一个自定义对象的封装。如果只是属于函数或算法,不是对象而且很多地方能用到,就放到Utility,比如排序/加密算法。


工程文件组织结构

关于网络通信模块的设计,我个人认为应该是每一个HTTP请求都应该独立互不干扰,就是你封装的POST/GET方法都会创建一个临时的对象,而长连接一般只维持一个实例对象采用单例的方式创建。我会为每一个HTTP 接口请求创建一个API对象,在里面请求数据,当然了为了避免请求代码的重复编写可以建立一个BASE API类,子类调用父类的请求方法就行了,不同的只是接口与参数。思想就是这样,以后有时间再来讲讲API类的设计。

关于多线程,在iOS开发中用得最多的就是GCD和NSOperation了,在大部分情况下GCD就够用了。但是NSOperation也有它的优势,比如可以设置并发个数,可以设置线程间的依赖,可以暂停和恢复,比较灵活,多线程下载就经常用它。面试中也经常会问GCD与NSOperation的区别,这个自己去查查,资料还是比较丰富的,底下也有篇关于NSOperation的参考链接。GCD虽然强大,可是还是有很多人瞎用,真替他们着急,随便说几点:

1.滥用dispatch_after做定时器导致crash!不知道还有个东西叫dispatch_source_set_timer吗?

倒计时

2.不要轻易用dispatch_sync,线程同步方法那么多为何你偏偏要爱上不该爱的它!尤其不要在dispatch_async里面用dispatch_sync,不要问为什么,百度里面google一下!

3.dispatch_once也就创建单例的时候用用,不要瞎用。dispatch_once是整个app生命周期内只执行一次,不是对象生命周期内只执行一次,大哥!

4.什么!你居然不知道用GCD创建串行线程与并行线程!

串行
并行

关于多团队协作,如果项目用分模块的方式进行开发,CocoaPods无疑是个非常好的工具。相对独立的模块尽量打成私有pod,这样,公共平台把整个项目的框架打好,其他的业务部门只需要自己建立一个私有pod,使用公共平台打好的那些私有pod独立开发调试就好,而不必时时提交代码到主工程,这样的话会非常麻烦,比如代码merge冲突,证书冲突,会疯掉的。当模块开发完毕再提交到主工程就好了。当然私有pod打多了也麻烦,私有pod依赖更多的私有pod在添加文件到target的时候会很痛苦,Xcode默认是target全部选上的,要跟Apple反馈一下希望他们会解决。(*Development Pods*里面添加文件需要选择target,默认都是选中的,而我们只需要添加到1个target里面去,依赖太多别的库就会有很多target。不过没关系,不用取消选择。添加完文件后执行一下pod install/pod update就行了,不用纠结那些选中的target。)

关于数据持久化,最重要的就是保持数据源的一致。还有一个比较重要的就是APP升级之后的向前兼容,那种你一升级app启动崩溃的问题,多半是新版本的数据结构发生变化,不支持老版本产生的数据或者设置等。我一直偏爱sqlite,用得最多的就是FMDB,对CoreData无爱。要知道苹果自己的app NEWS用的就是FMDB啊,我还记得有人问facebook的工程师“为啥你们的app速度慢呢”,“because we use core data!”。有个开源库MagicRecord,对CoreData的深度封装,我用过,其实也还不错。

另外,单元测试真的有必要吗?单元测试可以使逻辑清晰化,当你仅仅阅读单元测试代码的话,你会发现它们写的好像编程教科书里的伪代码。当TDD的时候,这一点尤其有用。通过写单元测试,你可以很快的把逻辑梳理清楚,然后用代码来实现它。单元测试可以降低代码的耦合度。我们知道,耦合度高的代码很难做单元测试,反过来,如果你必须做单元测试,你是不会把代码写的耦合度很高的。单元测试可以让你知道你对代码的修改是否影响到了原来就有的功能,但是这也是所有的回归测试都可以做的。单元测试的特点在于:它测试的东西足够小从而在代码重构后仍能复用。单元测试是唯一一个可能使覆盖率达到100%的测试。单元测试开始难,持续做的话会越来越容易,因为难主要是因为环境的搭建和桩函数的缺失。单元测试很容易定位Bug,它好像在你的程序中打了无数的断点。单元测试很费时间,不过,后续改Bug更费时间。

讲了这么多,最后说说需求吧。我认为在需求评审时应该尽量抛出细节问题给pm,他们不懂技术,如果需求不确定就盲目开发,然后中途再改需求,这是很伤士气的,尤其是涉及到架构的修改,这让我想到那张pm改需求被程序员拍板砖的图,嘿!需求不稳定,就别动工,宁愿打酱油看博客。好了,啰嗦了这么多,有多少人会看到这里呢?希望以后有空回头来增加些内容。


更多请参考:

iOS应用架构谈

说说ReactiveCocoa 2

iOS 并发编程之 Operation Queues

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

推荐阅读更多精彩内容

  • 作别些许的日子 不知你是否还会记得 那日争吵后彼此冷冷的沉默 是我说我走 是我说永不再联系 原是无心赌气的话赶话 ...
    柚宝妈咪阅读 310评论 2 3
  • 绿水鹤青山, 薄雾入眸眼。 思绪如潮涌, 感悟千声叹!
    文人墨客一湘竹书香阅读 239评论 0 0
  • 生个娃傻三年,你信么? 生两个娃呢?到底是三加三还是三乘三。 我反正是三的三次方。
    慧悦汇爱阅读 372评论 0 0
  • 只是三天我便明白了你,你不是像以前的正正之人,你虽然随口都在说你的孤独,可是你却不寻思去改变,你只想一直这样去远离...
    那天雨看见了雪阅读 218评论 0 1
  • 再次见到林微微的时候,我把她爸爸过世的消息告诉了她。出乎我意料的是,这个看起来很柔弱的女孩子居然没有哭,只是红着眼...
    _桃止阅读 267评论 3 4