标签:
感谢作者和译者
本文翻译自 futurice 公司的 iOS Good Practices,译文在 Github 上进行维护,同时在简书 上进行发布。
本文发出几天后发现网上也有了另外一个翻译版本:http://ios.jobbole.com/81830/
原标题是iOS Good Practices,应该翻译成 iOS 良好实践/优秀实践的,不过好拗口,而且已经发出去了,暂且就这样吧。
本文尚未经过大牛审校,纰漏瑕疵以及语句不顺的地方,以在 Github 提出,请大家多多指正。
以下是正文
就像软件一样,如果我们不持续改进这份文档,它就会落伍。我们希望大家都来帮助我们改进它 —— 只要开一个 issue 或者发送一个 pull request!
对其他平台感兴趣?看看我们的 Android 开发最佳实践 和 Windows App 开发最佳实践吧。
跳进了 iOS 的坑真是麻烦。无论是 Swift 还是 Objective-C, 都没有在其他地方广泛使用,而且这个平台对每个东西都几乎有它自己的命名方式,并且连要在真机上调试都充满了坎坷。无论你是刚刚入门 Cocoa 还是想纠正自己开发习惯的开发者,都能从本文档获益。不过下面写的仅仅是建议,所以如果你有一个更好的方案,那就试试吧!
Xcode 是大多数 iOS 开发者的选择,并且是 Apple 唯一官方支持的IDE。有一些其他的选择,比如 AppCode 是最有名的,但是除非你是经验丰富的开发者,否则就使用 Xcode 吧。不要管它的一些小缺点啦,它现在已经蛮好用咯。
要安装 Xcode,只需要下载 Mac App Store 中的 Xcode 。它会同时下载最新的 SDK 和模拟器,同时你可以从 Preferences > Downloads 下载更多的内容。
开始iOS开发的时候,一个常见的问题是用代码写所有的 view 还是使用 Interface Builder(Storyboards 或者 XIB )。两种方案都久经考量。但是,有下面几个考虑:
当把项目放入版本控制系统的时候,首先应该有一个好的 .gitignore
文件。这样,不必要的文件(用户设置,临时文件这些)都不会放进你的仓库里面。幸运的是,Github 已经给了我们 Objective-C 和 Swift 语言的模板
如果你计划增加外部依赖(比如,第三方库)在你的项目中,CocoaPods 提供了一个快捷的途径,就像这样:
sudo gem install cocoapods
要开始使用,仅仅需要在你的 iOS 项目目录下运行:
pod init
它会创建一个 Podfile, 会管理你所有的依赖,在 Podfile 中加入你的依赖后,允许
pod install
来安装第三方库并且将它们作为 workspace 的一部分,你的 workspace 也会包含你自己的项目。 一般推荐提交你自己的项目的依赖,而不是每个开发者在一个 checkout之后运行 pod install
。
注意在之后,你需要打开 .xcworkspace
而不是 .xcproject
,否则你的代码就不能被编译了,命令:
pod update
会升级所有的 pod 到最新版本,你可以用大量 符号 来定义你期望的版本需求。
为了组织目录里面的上百个源代码文件,最好根据你的架构来设置一些文件结构。比如,你可以用这样的:
├─ Models
├─ Views
├─ Controllers
├─ Stores
├─ Helpers
首先,将他们创建为 Group(黄色的目录),用 Xcode 的项目导航里面的你的项目中。然后对每个项目里的文件,将它么连接到真实的文件目录 —— 通过打开它右边的文件检查器,点击小小的目录图标,在你的项目目录下创建一个和 group 同名的子目录。
一开始就应该把所有的显示给用户的字符串放进本地化文件。这样不仅仅为了翻译方便,同时也便于查找用户看见的文本。你可以在 build Scheme 中加入一个启动参数来指定特定的语言
-AppleLanguages (Finnish)
对于更复杂的翻译问题,比如复数(比如 "1 person" 和 "3 people"),你应该使用 .stringsdict
format 而不是一个普通的 localizable.strings
文件。在有了这个强大的工具来处理比如 "one", some", "few" 和 "many" 的情况。 当处理俄罗斯语和阿拉伯语的时候,你就不用急得抓耳挠腮了。
关于本地化的更多信息,看 2012年 2月的 Helsink iOS 会议的 幻灯片,大部分内容对现在也是适用的。
创建被 prefix header 引入的一个 Constants.h
文件。
不要用宏定义(用 #define
),用实际的常量定义
static CGFloat const XYZBrandingFontSizeSmall = 12.0f;
static NSString * const XYZAwesomenessDeliveredNotificationName = @"foo";
常量类型安全,并且有更明确的作用域(并不是在所有没有被引入的文件中能使用)。不能被重定义,并且可以在调试器中使用。
特别是分发你的 app 的时候(如提交到 App Store),最好把分支用特别的 tag 区分。同时,新特性的开发,会引入很多 commit 的,也应该在独立的分支上提交。git-flow
是一个帮助你遵从这个约定的工具。它简单地包装了 git的分支和tag的命令,帮助维护一个正确的分支结构。所有的开发分支(比如 develop
), 关于 app版本的tag分支以及提交到 master分支的:
git flow release finish <version>
总得来说,来把一个第三方库加入到你的项目中需要慎重考虑。确实,一个灵巧的库或许能帮助你解决现在的问题,但是可能在之后陷入维护的噩梦,比如在下一个系统版本改变一些东西之后,或者一个第三方库的场景变成了官方的API。不过在一个良好的设计的代码中,切换实现是很轻松的。尽量考虑用 Apple 的广泛(而且优秀的)框架吧~
这个小节尽量保持剪短。库的特性是为了减少模板代码(比如 AutoLayout)或者解决复杂的需要很多测试的问题,比如日期计算。当你在 iOS 中更加专业的时候,挖掘这里的源代码,通过它们底层的 Apple 框架来认识他们,你会发现这些可以做大量工作。
99.95% 的 iOS开发者使用这个网络库,当 NSURLSession
自己本身也非常完善的时候,AFNetworking
仍然能凭借很多 app 需要的队列请求管理功能立足于不败之地。
总得来说, 不要自己写日期计算.幸运的是,有一个 DateTools 是 MIT 协议的,而且经过彻底测试的,你可以放心的在需要用日期的时候使用它。
如果你更喜欢在代码里面写界面,你会用过 Apple 难用的 ‘NSLayoutConstraint‘ 的工厂或者 Visual Format Language 。前者很罗嗦,后者基于字符串,不利于编译器检查。
Masonry 通过它们自己的 DSL 来取代常量, Swift 中一个类似的库是 Cartography,它利用了语言的丰富的操作符重载特性。如果更加保守的话,FLKAutoLayout 提供了一个干净但是没有太多魔法的原生 API 包装。
RACSignal
s 或者 void
返回值的自定义 block 方法来返回。这里有一些通知其他对象的常用方法:
保持你的 model 是不可变的,以及用他们来改变远程的 API 的语义以及类型。Github 的 Mantle 是一个好的选择
当用 Auto Layout 布局你的 view 的时候,确保在你父类中加入了下面的代码:
+ (BOOL)requiresConstraintBasedLayout
{
return YES;
}
否则当系统没有调用 -updateConstraints
的时候,你可能会遇到奇怪的 bug。
建议使用依赖注入,比如:传递任何需要的对象作为参数,而不是在一个单例中保持所有的状态。后一种方法仅仅在状态是 真的 全局的时候适用。
+ [[FooDetailsViewController alloc] initWithFoo:(Foo *)foo];
// GigStore.h
typedef void (^FetchGigsBlock)(NSArray *gigs, NSError *error);
- (void)fetchGigsForArtist:(Artist *)artist completion:(FetchGigsBlock)completion
// GigsViewController.m
[[GigStore sharedStore] fetchGigsForArtist:artist completion:^(NSArray *gigs, NSError *error) {
if (!error) {
// Do something with gigs
}
else {
// :(
}
];
这样运行,但是如果你有多个组合的网络请求的时候,就会进入回调嵌套的地狱。
如果你发现已经陷入了回调的地狱,看看 ReactiveCocoa (RAC) 吧,它是一个万能而且多功能的库,能改变大家写 整个 app 的方法,但是你可以在它适用的地方保守地使用它。
在 Teehan+Lax 和 NSHipster 上面有一些关于 RAC (and FRP in general) 以及函数式编程的概念的优秀介绍。
// GigStore.h
- (RACSignal *)gigsForArtist:(Artist *)artist;
// GigsViewController.m
[[GigStore sharedStore] gigsForArtist:artist]
subscribeNext:^(NSArray *gigs) {
// Do something with gigs
} error:^(NSError *error) {
// :(
}
];
它允许通过信号和其他信号的结合,在展示它们之前做一些改变。
Asset catalogs 是管理你所有项目可视化资源的最好方式,它们可以同事管理通用的以及设备相关的iPhone 4-inch, iPhone Retina, iPad,等)资源,并且会自动通过它们的名字分组。告诉你的设计师如何添加它们(Xcode有内置的 Git 客户端)可以节省很多时间,否则你会话很多时间来在从邮件或者其他渠道复制到代码库里面。它可以让设计师马上尝试资源改变的样子,并且反复实验。
Asset catalogs 仅仅暴露了图片的名称,图片集里面的抽象的名字。这可以避免资源名字的冲突,就像 button_large@2x.png
的文件的命名空间在它的图片集里面。遵守一些命名规则可以让生活更美好:
IconCheckmarkHighlighted.png // Universal, non-Retina
IconCheckmarkHighlighted@2x.png // Universal, Retina
IconCheckmarkHighlighted~iphone.png // iPhone, non-Retina
IconCheckmarkHighlighted@2x~iphone.png // iPhone, Retina
IconCheckmarkHighlighted-568h@2x~iphone.png // iPhone, Retina, 4-inch
IconCheckmarkHighlighted~ipad.png // iPad, non-Retina
IconCheckmarkHighlighted@2x~ipad.png // iPad, Retina
修饰后缀 -568h
, @2x
, ~iphone
and ~ipad
是不必要的,但是有他们在文件里面,当把文件拖进去的时候,Xcode会正确地处置它们。这避免赋值错误。
你可以用设计师原始的 vector graphics (PDFs) 加入到 asset catalogs,Xcode可以自动地根据它们生成位图。这减少了你的工程的复杂性(管理更少的文件)。
尽管命名约定很长,但是Apple一如既往地在 API中 遵守了命名原则。
这里有一些你应该使用的基本原则:
一个方法用 动词 开头的表示它做了一些副作用,但是不会返回任何东西:- (void)loadView;
- (void)startAnimating;
任何以 名词 开头的方法,没有副作用而且返回了它指的对象:- (UINavigationItem *)navigationItem;
+ (UILabel *)labelWithText:(NSString *)text;
尽量分离两者,比如,不要在操作数据的时候产生副作用,反之亦然。这会让你的副作用包含到代码更小的细分粒度里面,让调试变得非常困难。
Pragma marks 是把你代码分组的一个好方法,特别是在 view Controller里。下面的结构可以适用于绝大多数 view Controller的工作:
#import "SomeModel.h"
#import "SomeView.h"
#import "SomeController.h"
#import "SomeStore.h"
#import "SomeHelper.h"
#import <SomeExternalLibrary/SomeExternalLibraryHeader.h>
static NSString * const XYZFooStringConstant = @"FoobarConstant";
static CGFloat const XYZFooFloatConstant = 1234.5;
@interface XYZFooViewController () <XYZBarDelegate>
@property (nonatomic, copy, readonly) Foo *foo;
@end
@implementation XYZFooViewController
#pragma mark - Lifecycle
- (instancetype)initWithFoo:(Foo *)foo;
- (void)dealloc;
#pragma mark - View Lifecycle
- (void)viewDidLoad;
- (void)viewWillAppear:(BOOL)animated;
#pragma mark - Layout
- (void)makeViewConstraints;
#pragma mark - Public Interface
- (void)startFooing;
- (void)stopFooing;
#pragma mark - User Interaction
- (void)foobarButtonTapped;
#pragma mark - XYZFoobarDelegate
- (void)foobar:(Foobar *)foobar didSomethingWithFoo:(Foo *)foo;
#pragma mark - Internal Helpers
- (NSString *)displayNameForFoo:(Foo *)foo;
@end
最重要的一点是在你项目的类中保持一致性
我们公司没有任何公司级别的代码风格指南,详细看看其他开发者的 Objective-C 风格指南很有用,即使一些内容是公司相关的或者过于激进了。
推荐你尽可能多打开编译警告,并且像对待错误一样对待编译警告。推荐 这个PPT。这个幻灯片覆盖了如何在特定文件,或者特别代码段里面消除相关警告的内容。
简单的来说,至少需要在 _“Other Warning Flags” 编译设置里面定义下面的值:
-Wall
_(增加很多的警告)_-Wextra
_(增加更多的警告)_同时打开 “Treat warnings as errors”
Clang 编译器(Xcode使用的)有一个 静态分析器 来进行你的代码控制和数据流的分析,来检测编译器不能检测的许多错误。
你可以通过在 Xcode 里面手动运行 Product → Analyze 菜单项来手动执行代码分析
分析器可以用浅或者深的模式允许,后者更加慢,但是可以从跨函数的控制流和数据流上分析更多问题
推荐:
我们自己的Ali Rantakari 创建的,Faux Pas 是一个极佳的静态错误检测工具,它分析你的代码并且找出那些你自己甚至都没发现的问题。在提交你的 App 到应用商店前用它吧!
当你的 App 崩溃的时候,Xcode 不会默认进入到调试器里面。为了调试,你需要增加一个异常断点(在 Xcode 的 Debug 导航中点 “+”),来在异常发生的时候退出执行。在很多情况下,你需要看看触发这些异常的代码。它会捕捉任何异常,即使是已经处理的。如果 Xcode 在 一个第三方库里面中断执行,比如,你可能需要通过选择 Edit Breakpoint 并且设置 Exception 为 Objective-C.。
对于视图 debug, Reveal 和 Spark Inspector 这两个强有力的可视化检查工具可以帮你省下很多时间,特别是在你使用 Auto Layout 并且希望定位出问题或者溢出屏幕的视图的时候。Xcode 提供了免费的类似功能 ,但是只能适用于 iOS 8+ 并且不那么好用。
Xcode 有一个叫 Instruments 的分析工具,它包括了
许多分析内存,CPU,网络通讯,图形以及更多的工具,它有点复杂的,但是它的追踪内存泄漏的时候还是蛮直观的。只需要在 Xcode 中 选择 Product > Profile,选择 Allocations, 点击 Record 按钮并且用一些有用的字符串过滤申请空间的信息,比如你自己的app的类名。它会在固定的列中统计,并且告诉你每个对象有多少实例。到底是什么类一直增加实例导致内存泄漏。
Instruments 也有自动化的工具来进行录制并且运行UI交互以及JavaScript文件。. UI Auto Monkey 是一个自动化随机点击、滑动以及旋转你的app的脚本,他在压力、渗透测试中很有用。
强烈推荐使用一些统计框架,他们直观地告诉你有多少人用你的应用。X 特性增加了用户么?Y 按钮很难找到么?为了回答这些问题,你可以发送事件,允许时间以及其他记录的信息到一个聚集它们并且可视化它们的服务。比如, Google Tag Manager 。这个是一个比 Google Analytics 更加有用的地方是可以在app 和统计之间插入数据,所以数据逻辑可以通过 web 服务进行修改,而不用更新你的app。
一个好的实践是创建一个简单的 helper 类,比如 XYZAnalyticsHelper
,处理 app 内部 model 以及数据格式 (XYZModel, NSTimeInterval, …)的变换,来适配字符串为主的数据层,
- (void)pushAddItemEventWithItem:(XYZItem *)item editMode:(XYZEditMode)editMode
{
NSString *editModeString = [self nameForEditMode:editMode];
[self pushToDataLayer:@{
@"event": "addItem",
@"itemIdentifier": item.identifier,
@"editMode": editModeString
}];
}
另外的优点是,你在必要的时候可以替换整个统计框架,而不用改变 app 其他部分。
应该让你的 app 向一个服务发送崩溃日志。你可以手动实现,通过 PLCrashReporter以及你自己的后端。但是强烈推荐你使用现有的服务,比如下面的
当你配置好后,确保你 保存了 the Xcode archive (.xcarchive
) 对于每一个 app 放出的版本。这个 归档中包含了构建的app的二进制以及调试符号(dSYM
),你需要用每个版本特定的app把你的 Crash 报告符号化。
每一个简单的 app 都可以不同的方式构建,最基本的分离是 Xcode 给你 debug 和release 之间的构建方案。后者在编译的时候有更多的优化,可能会导致你需要多调试一些问题。 Apple 建议你在开发的时候用 debug 模式,在打包的时候用 release 设置。这是默认的 Scheme (Play 和 Stop 后面的下拉菜单),运行Run 的时候会 用 debug设置而运行 Archive 的时候会使用 release。
然后,对于真实的 app 这似乎太简单了。你 不该 设置多个为测试,staging和其他相关的开发活动。每一个可能有自己的 URL,日志级别,bundle ID)所以你可以一起安装它们,以及描述文件。然后一个简单的 debug/release 区别不能分离这些,你可以在你项目的设置中的 Info 选项卡做更多的编译设置。
xcconfig
文件通常构建设置是 Xcode GUI定义的,但是你同样可以用 configuration settings files(“.xcconfig
files”),优点是:
#include
其他构建设置文件, 能帮助你减少重复:
Common.xcconfig
并且在其他构建设置文件里面 #include
#include "MyApp_Debug.xcconfig"
来重载其他设置更多关于这个主题的信息请看这个 PPT。
一个 target 是处于比项目更低一级的级别。比如,一个项目可能有多个target,可能重载它的项目设置。简单地说,每个 target 和一个 app相当。比如,你可能有几个因为国家区分的 app (从同样的代码编译)来提交到 App Store。每个会有 development/staging/release 的构建,所以最好通过构建设置而不是 target来区分。一个 app 只有一个 target 是很少见的。
Schemes 告诉 Xcode 在你点击 Run, Test, Profile, Analyze 或者 Archive 操作的时候应该怎么做。它们把这些操作映射到一个 target 和一个构建设置中。你可以传递启动参数,比如 app 需要允许的语言(为了测试本地化)或者一些了为了调试用的诊断标志。
一个建议的 Scheme 命名是 MyApp (<Language>) [Environment]
:
MyApp (English) [Development]
MyApp (German) [Development]
MyApp [Testing]
MyApp [Staging]
MyApp [App Store]
对于大多数环境来说语言部分是不必要的,app 会可能会以非 Xcode 的方式安装,比如用 TestFlight, 启动参数会被忽略。这个情况下,为了测试本地化需要手动设置设备的语言。
部署一个软件到 iOS 设备上并不直观。但是有一些核心观点,只要理解了,对你有很大的帮助。
当你需要在真实设备上运行软件的时候,你需要用一个 Apple 认证的 证书 签名。每一个证书是连接到一个 公、私 密钥对,私钥会保存在你的 Mac 的 KeyChain 里面,证书有两种类型
除了证书,还有 描述文件, 它把设备和证书连接起来。而且,它分成开发和发布两种类型。
想同步所有的证书和描述文件到你的机器,可以去 Xcode 的 Preferences 的 Accounts下,添加你的 Apple ID, 然后双击你的 Team 名称。然后底部会有一个刷新按钮,有时候你需要重启 Xcode 来让东西显示出来。
有时候你需要调试一个描述文件的问题。比如,XCode 可能拒绝安装 app 到一个设备中,因为设备没有在开发或者 Ad-doc发布的的描述文件设备列表中。这个情况下,你可以使用 Craig Hockenberry 的 Provisioning 插件,浏览 ~/Library/MobileDevice/Provisioning Profiles
,选择一个 .mobileprovision
文件,并用空格键调用 Finder 的快速查看特性,它会告诉你关于设备,entitlements,证书以及 Apple ID的信息。
iTunes Connect 是你管理上传到 App Store的 App 的后台网站。 Xcode 6 需要一个 Apple ID 作为开发者账号来签名并且上传二进制。如果你有多个开发者账号而且要上传它们的 app 就需要一点技巧。 因为 一个 Apple ID 只能和一个 iTunes Connect 账号关联, 一个变通方案是为每个 iTunes Connect 账号创建一个新的 Apple ID, 使用 Application Loader 取代 Xcode 来上传应用。这样有效地解除了上传最终 .app
文件构建和签名的耦合。
在上传二进制后,耐心等待,可能要花上一个小时。当你的 App 出现后,可以链接到对应的App版本并且提交审核
当验证一个 App内购的 receipt的时候,记住做以下步骤:
如果可能,把你的 IAP 存储销售相关的内容存储在服务器端,并且只在一个合法的经过上述检查的 receipt。这样的设计避免了常见的盗窃机制,同时,因为服务器做了验证,所以你可以使用 Apple 的 HTTP 验证服务来取代你自己的 PKCS #7
/ ASN.1
格式。
更多关于这个主题的信息可以看Futurice blog: Validating in-app purchases in your iOS app
标签:
原文地址:http://www.cnblogs.com/code-changeworld/p/4749124.html