IOS 开发规范

公司名称 蔡令个人博客有限公司 文档编号
文档名称
起 草 蔡令
审 批

修订历史
版本号 日期 状态 修订人 摘要
1.0 2020/4/22 C 蔡令 起草
状态标识:C –Created A- Added M - Modified D - Deleted

阅读指南
单位/人员 建议阅读内容

目录
一.命名规范 4
1.1变量与常量 4
1.2类、分类、协议、对象、block 5

1.3属性 6
1.4方法 6
1.5通知和异常 7
1.6资源命名 7
1.7命名建议 7
1.8Constant命名规范 7
二.格式规范 8
2.0编码代码的块基本规范 8
2.1空格 9
2.1.1属性相关 9
2.1.2方法相关 9
2.1.3语句 9
2.1.4容器 9
2.2 空行 10
2.4 折行 10
2.5 大括号{} 10
三.控制规范 12
3.1switch 12
3.2if-else 12
四.内存管理规范 14
4.1 注释方式 14
4.3延迟调用范围 15
4.4类型设计范围 15
五.代码组织规范 16
5.1地址参考 16
5.2VIEWCONTROLLER 控制器代码规格 16
六. 注意要点 16
七. 工程项目结构规范 17

一.命名规范
1.1变量与常量
1).局部变量:
1.【强制】局部变量不应该包含下划线,尽量避免单个字符变量,力争⻅名知义。
√ 正 例 : CGFloat cellHeight = 22;
X 反例: NSString a = @"This is test..."; X 反例: NSInteger _index = 10;
2).成员变量:
1.【强制】以空格键(下划线)开头,小驼峰命名。
√ 正例: UIButton
_myInstanceVariable; X 反例: UIView coverView;
3).常量:
1.【强制】以小写 k 开头,使用驼峰式命名规则。
2.【建议】使用 const 来实现常量,不建议使用 define 宏定义方式
√ 正例:
static int const kNumberOfFiles = 50;
static NSString
const kUserKey = @"yixin";
X 反例:
int MAXVALUE = 200;
#define CompanyName @"RayWenderlich.com" #define thumbnailHeight 2
4).枚举:
1.【强制】枚举命名要添加相关的类名前缀,驼峰命名,并且枚举值命名要加类型前缀。
2.【建议】建议使用 typedef NS ENUM() 形式,不用 C 定义方式。
√ 正例:
typedef NS_ENUM(NSInteger, UIViewAnimationTransition) { UIViewAnimationTransitionNone, UIViewAnimationTransitionFlipFromLeft, UIViewAnimationTransitionFlipFromRight, UIViewAnimationTransitionCurlUp, UIViewAnimationTransitionCurlDown,
};
X 反例:
// 命 名
typedef NS_ENUM(NSUInteger, YCACCOUNTTYPE) { YCACCOUNTTYPE_CUSTOM, YCACCOUNTTYPE_MAIL, YCACCOUNTTYPE_TEL,
};

// C⻛格typedef enum : {
CameraModeFront, CameraModeLeft, CameraModeRight,
} CameraMode; 5). 宏定义:
宏定义使用方式分两种⻛格:常量与工具宏,常量宏使用全大写字⺟,由下划线(_)分割;工具宏使用全大写字⺟形式。示例如下:
#define CLS HAS LOAD METHOD //定义常量
#define ISSELECTOR(sel) sel_isMapped(sel) // 定义一个工具
1.2类、分类、协议、对象、block
1).类:
1.【强制】类名以大写字⺟开头,应该包含一个名词来表示它代表的对象类
2.【建议】同时可以加上必要的前缀,为表达清楚、减少重名,建议采用:前缀 + 模块 + 功能 + 类型的命名方
式。
√ 正例:NSString
√ 正例:YXHomeBannerView
X 反例:HeaderView (不明确,不能⻅名知意) 2). 分类:
1.【建议】分类需要能够通过名称理解内部提供的方法是什么功能。
2.【建议】不要用官方框架/类库的名称前缀,最好前缀 3 个字符⻓。
// 建 议
@interface UIViewController (YXMediaPlaying) @interface NSString (YXStringEncodingDetection)
// 不建议
@interface NYTAdvertisement (private) @interface NSString (NYTAdditions)
3).协议:
1.【建议】协议名称应该清晰地表示它所执行的行为,而且要和类名区别开来。说明:协议通常使用 ing 词尾来命名一个协议,比如 NSCopying,NSLocking。
2.【建议】有些协议本身包含了很多不相关的功能,主要用来为某一特定类服务,这时候可以直接用类名来命名这
个协议,比如 NSObject 协议,它包含了 id 对象在生存周期内的一系列方法。
√ 正例:NSCopying、 NSObject 4). 对象:
1.【建议】对象命名时建议采用 修饰/义务名称+类型。如:goodsNameLabel
说明:如果只用修饰命名会引起歧义, 比如叫:goodsName,这个到底是个 NSString 还是 UILabel? 同样的, 如果只用类型来命名则会缺失作用信息, 比如叫:label, 不知道具体功能。
5). block
block 的命名采用:前缀+名称+Block 的形式,名称采用驼峰命名,示例如下:
typedef void (^AFNetworkReachabilityStatusBlock)(AFNetworkReachabilityStatus statu s);

1.3属性
1).【强制】属性命名,采用小驼峰命名,勿用下划线开头(PS:默认 XCode 会自动生成与之对应的下划线为前缀的成
员变量)。
2).【强制】不要使用 new 开头。
3).【强制】不要带有 is 前缀,布尔类型可以复写 getter 方法命名。
4).【强制】不要带有 get 前缀, getter 方法会自动添加。
说明:如果涉及到接口字段命名不符或不规范,请使用模型与字典转换工具(YYModel、MJExtension 等) 提供的modelCustomPropertyMapper 等类似方法来摸映射.
√ 正例:
@property (nonatomic, copy) NSString descriptiveVarialbleName; @property (assign, getter=isEditable) BOOL editable;
X 反例:
@property (nonatomic, copy) NSString
User_MaxAge;
1.4方法
1).基本规约:
对于方法的命名,遵守以下的命名基本规范:
1.【强制】小写字⺟开头,多参数方法分割段首字⺟小写。
2.【强制】如该方法返回接收者的属性,则在属性后命名该方法。不必要使用 get 开头。
3.【建议】对表示对象所采取的动作的方法,采用动词开头,尽量不要用 do / does 等普遍词。
√ 正例:
-(void)invokeWithTarget:(id)target object:(id)obj;
-(void)saveSearchHistory;
-(CGSize)cellSize;
-(id)viewWithTag:(NSInteger)aTag;
-(int)runModalForDirectory:(NSString )path file:(NSString )name types:(NSArray)fileTypes; X 反例:
-(void)SelectItem:(id)tabViewItem AtIndex:(int)index; //(大小写)
-(void)history; //(没有说清,表示动作要添加动词)
-(CGSize)getCellSize; //(没必要用 get)
-(id)taggedView:(int)aTag; //(aTag 的具体释义不明确)
-(int)runModalForDirectory:(NSString
)path andFile:(NSString )name andTypes:(NS Array )fileTypes; //(没必要用 and)
2).代理方法:
1.【强制】代理方法命名以标示发送消息的对象的类名开头,省略类名的前缀并小写首字⺟。
√ 正例:
-(NSInteger)tableView:(UITableView )tableView numberOfRowsInSection:(NSInteger)section; X 反例:
-(void)clickedBtn:(UIButton
)btn;
2.【强制】至少保留代理的源对象参数tableView,示例如下。
-(NSInteger)numberOfSectionsInTableView:(UITableView *)tableView; 3.【建议】使用 Did 或 will 词来对代理方法执行前后状态进行通知
√ 正例:

-(void)browserDidScroll:(NSBrowser )sender;
-(NSUndoManager
)windowWillReturnUndoManager:(NSWindow )window;
4.【建议】使用 "should" 来命名代理方法,以询问是否允许执行某项操作或事件。
√ 正例:
-(BOOL)textFieldShouldBeginEditing:(UITextField
)textField;
1.5通知和异常
1).【建议】通知的时机分先后,因此在方法命名的时候要有相关表明时态的字段。如: will / did 等。常规的通知命名
格式为:
[前缀/模块名称] + [Did | Will ] + [唯一单位名称/事件] + NSNotification
√ 正例:
NSApplicationDidBecomeActiveNotification NSWindowDidMiniaturizeNotification NSTextViewDidChangeSelectionNotification NSColorPanelColorDidChangeNotification
2).【建议】异常命名和通知有一些类似,常规的命名方式为: [前缀] + [唯一的单位名称] + NSException
√ 正 例 : NSColorListIOException
NSColorListNotEditableException NSDraggingException NSFontUnavailableException NSIllegalSelectorException

1.6资源命名
1).图片资源.
1.【建议】建议根据 模块_功能/作用/内容_样式 来分割命名。如:[email protected]
2).文件资源
1.【建议】建议使用驼峰命名文件资源。如:CityList.json
1.7命名建议
1).【建议】命名尽量使用常规的单词,避免生僻不易懂的释义。示例:
read 读 取 /write 写 入load 载 入 /save 保 存 , encode 编码 /decode 解码
encrypt 加密 /decrypt 解密,
1.8Constant 命名规范
1).枚举常量
【必须】使用枚举类型来表示一组相关的整型常量。
【建议】枚举常量和typedef 定义的枚举类型的命名规范同函数的命名规范一致。(参考 Naming Functions) typedef enum _NSMatrixMode {

NSRadioModeMatrix = 0,
NSHighlightModeMatrix = 1,
NSListModeMatrix = 2,
NSTrackModeMatrix = 3
} NSMatrixMode;
注意:上面枚举typeof 中的_NSMatrixMode 是无用的。我们可以像位掩码(bit masks)一样创建一个匿名枚举,如下enum {
NSBorderlessWindowMask = 0,
NSTitledWindowMask = 1 << 0,
NSClosableWindowMask = 1 << 1,
NSMiniaturizableWindowMask = 1 << 2,
NSResizableWindowMask = 1 << 3
};
2).使用const 关键字创建常量
【必须】使用 const 关键字创建浮点型常量。你也可以使用 const 来创建和其他常量不相关的整型常量。否则,请使用枚举类型来创建。即,如果一个整型常量和其他常量不相关,可以使用 const 来创建,否则,使用枚举类型表示一组相关的整型常量。以下例子声明了 const 常量的格式:
const float NSLightGray; 3).其他常量类型
【必须】通常情况下,不要使用#define 预处理命令(preprocessor command)创建常量。正如上面所说,对于整型常量, 使用枚举创建;对于浮点型常量,使用 const 修饰符创建。
【必须】有些符号需要使用大写字⺟标识。预处理器需要根据这个符号进行计算以便决定是否要对某一块代码进行处理。比如:
#ifdef DEBUG
注意:那些编译器定义的宏,左侧和右侧各有两个下划线。如下:
MACH
【必须】通知的名字和字典的 key,应该使用字符串常量来定义。使用字符串常量编译器可以进行检查,这样可以避免拼写错误。Cocoa 系统库提供了许多字符串常量的例子,比如:
APPKIT_EXTERN NSString NSPrintCopies;
字符串常量应该在.h 头文件中暴露给外部,而字符串常量真正的赋值是在.m 文件中。如下:
.h 文件
extern NSString
const WSNetworkReachablityStatusDidChangedNotification;
.m 文件
NSString * const WSNetworkReachablityStatusDidChangedNotification = @"WSNetworkReachablityStatusDidChangedNotification";

二.格式规范
2.0编码代码的块基本规范
1).【建议】建议行字符数设置为:100
2).【建议】单个类的代码行数尽量不超过 300 行

3).【建议】单个方法实现尽量不超过 30 行,有超过的要想办法分解。
2.1空格
2.1.1属性相关
1).[强制] @property 后留一空格。
2).[强制] ()里面,逗号紧跟前一变量,与后一变量之间留一个空格。
3).[强制] ()外面,先留一个空格,再声明属性。
4).[强制] 类型与属性名之间留一空格,紧跟属性名。
√ 正例:
@property(nonatomic,readonly,copy)NSString
appPath; X 反例:
@property(nonatomic,copy)NSString *appPath;

2.1.2方法相关
1).【强制】方法类型(+/-符号)之后有一空格。
2).【强制】在方法各段之间有一个空格 。
3).【强制】在方法实现的左大括号和方法名之间有一个空格。
4).【强制】方法参数类型和
之间有一个空格
√ 正例:

  • (UITableViewCell )tableView:(UITableView )tableView cellForRowAtIndexPath:(NSI ndexPath *)indexPath;
    2.1.3语句
    1).【强制】等号 = 或 == 前后需要加空格。
    2).【强制】if-else 与后面 { 之间有一个空格 。
    √ 正例:
    NSInteger index = 10; if (xxx) {
    //..
    }else{
    //...
    }

2.1.4容器
1).【建议】字面量数组元素之间,逗号后面有空格。
2).【建议】字面量字典 key 与 value 之间,冒号前后有空格。
√ 正例:
NSArray array = @[[foo description], @"Another String", [bar description]]; NSDictionary dict = @{NSForegroundColorAttributeName : [NSColor redColor]}; 3).【建议】在写一行不太优雅的情况,可以采用如下形式。
√ 正例:
NSArray array = @[@"This",@"is",@"an", @"array"];
NSDictionary
dictionary = @{NSFontAttributeName : [NSFont fontWithName:@"Helvetica-Bold" size:12], NSForegroundColorAttributeName : fontColor};

2.2空行
1).【强制】方法实现之间空一行。
2).【强制】@interface 和 @implementation 与 "#import" 之间空一行
3).【强制】使用 #pragma mark - 来分区代码块 上下各空一行。之间空一行。4).【强制】 @synthesize 和 @dynamic 在实现中每个都应该新占一行。
√ 正例:
@implementation YXTestClass #pragma mark - Public
-(void)PublicMethod{
//...
}
#pragma mark - Private
-(void)PrivateMethod{
//...
}
#pragma mark - UITableViewDelegate

@end
2.3对齐
1).[强制] 方法调用中,应该将所有参数写在一行,如 示例 1 .
2).[强制] 如果参数比较多或方法比较⻓,采用 示例 2 ⻛格.
3).[强制] 不要使用 示例 3 ⻛格.
√ 正例:
示例 1(正确):
[bird singSong:@"HaHa" atTime:@"7am" inThe:@"sky"]; 示例 2(正确):
[myObject doFooWith:arg1 name:arg2 error:arg3];
[myObj short:arg1 longKeyword:arg2 evenLongerKeyword:arg3 error:arg4]; X 反例:
示例 3(错误):
[myObject doFooWith:arg1 name:arg2 error:arg3]; // 参数大于 1 啦[myObject doFooWith:arg1 name:arg2 error:arg3]; // 参数大于 1 啦[myObject doFooWith:arg1 name:arg2 error:arg3]; // 按冒号垂直对⻬
2.4折行
一些时候单行不足以显示全部内容,可进行折一行,空 2 个空格显示。示例:
self.productsRequest = [[SKProductsRequest alloc] initWithProductIdentifiers:productIdentifiers]; self.productsRequest = [[SKProductsRequest alloc]
initWithProductIdentifiers:productIdentifiers];
2.5大括号{}
1).【强制】方法的大括号独占一行。
X 反例:

  • (NSString *)degreeWithScore:(NSInteger)score {

...
}
√ 正例:

  • (instancetype)initWithFrame:(CGRect)frame
    {
    ...
    }
    2).【强制】if/else/else if/while/for/do/block 等的大括号和相关内容在一行,前后一个空格分割。if 示例:
    if (a < 10){
    ...
    }
    if else 示例: if (a < 10){
    ...
    }esle{
    ...
    }
    if else-if else 示 例 : if (a > 10){
    ...
    }else if (a < 10){
    ...
    }else{
    ...
    }
    while 示 例 : while (a < 100){
    ...
    }
    for 示例:
    for (int i = 0; i < 100; i++){
    ...
    }
    for-in 示例:
    for (NSString *obj in arr){
    ...
    }
    block 示例:
    NetworkSuccessBlock theSueccessBlock = ^(id data){
    ...

}
do 示例:
do {

} while (a > xxx){
..
}

三.控制规范
3.1switch
1.在一个 switch 块内,每个 case 要么通过 break/return 等来终止,要么注释说明程序将继续执行到哪一个 case
为止,都必须包含一个 default 语句并且放到最后,即使它的代码块为空。
3.2if-else
1).[强制] if/else/for/while/do 语句中必须使用大括号,即使只有一行代码,避免如下形式。
X 反例:
if (condition) statements;
√ 正例:
if (condition) { statements;
}
2).[强制] if else if()... 的方式勿超过 3 层,使用 [卫语句] 或 [状态模式] 代替. [卫语句]把复杂的条件表达式转换为多个if 语句,实现它的逻辑,简化了逻辑。
X 反例:
if( 条 件 1){ printf("Error,1");
} else if( 条 件 2){ printf("Error,2");
} else if(条件 3) { printf("Error,2");
} else { printf("Error,2");
}
√ 正例: if(条件 1) {
printf("Error,1"); return;
}
if( 条 件 2) { printf("Error,2"); return;
}
if( 条 件 3) { printf("Error,2"); return;

}
printf("Error,2");
3).[强制] 不要在条件判断中执行其它复杂的语句,将复杂逻辑判断的结果赋值给一个有意义的布尔变量名,以提高可读性。
X 反例:
if (([str isEqualToString:@""] || [str containsString:@""]) && str.length > 10) {
// ...
}
√ 正例:
BOOL validate = ([str isEqualToString:@""] || [str containsString:@""]) && str.len gth > 10; if (validate) {
// ...
}
4).[强制] 不要将 BOOL 类型变量直接和 YES 比较
说明:BOOL 在 Objective-C 中被定义为 signed char 类型,这意味着一个 BOOL 类型的变量,不仅仅可以表示YES(1) 和 NO(0) 两个值,所以永远不要将 BOOL 类型变量直接和 YES 比较。
// 错误,无法确定 |great| 的值是否是 YES(1) BOOL great = [foo isGreat]; if (great == YES) {
}
// 正 确
BOOL great = [foo isGreat]; if (great) {
}
5).[强制] 不要使用诸如 nil == Object 或 Object == nil 或 someCondition == nil 的形式来判断。
说明:在 objc.h 文件中可⻅,nil / Nil 是 DARWIN_NULL 的宏定义,在条件语句中,会被当做 0 或者 false。在 C 源语言中,条件语句中 someCondition 值是 -1 和 true 都会被定义为 1 ,然而 someCondition == true 则可能为 false .
// 正确,直接判断if (!objc) {
...
}
// 错误,不要使用 nil == Object 的形式if (nil == objc) {
...
}
6).[建议] 循环体中的语句要考量性能。定义对象、变量、异常处理等操作尽量移至循环体外处理。
7).[建议] 双目运算符,只有当它可以增加代码清晰度或整洁时才使用。单一的条件都应该优先考虑使用。多条件时通常使用 if 语句会更易懂,或者重构为实例变量。
推荐:
result = a > b ? x : y; 反对:
result = a > b ? x = c > d ? c : d : y;

四.内存管理规范
4.1注释方式
1).单行注释:双斜杠后面空格,加上相关的内容说明。
/// ..
2).多行注释:使用斜杠和星号来进行多行注释,首行 2 个*.
/*
...
/

4.2注释范围
1).[强制] 对于每一个类,必须在头文件中为其添加类注释,说明类的职能,使用[多行注释]⻛格
/*
⻋型对比圆形浮窗视图
/
@interface YXCarCompareFloatView : UIView

@end
2).[强制] 对于类的共有属性,必须在头文件中为其添加属性注释,说明作用、默认值等,使用[单行注释]方式。
/// 月供标签,展示月供金额,须客户端拼接货币单位:元
@property (nonatomic, strong) UILabel *monthlyPaymentLabel;
3).[强制] 对类的共有方法,应在头文件为其添加注释,说明作用、参数、返回值等,使用[多行注释]⻛格。
/*
构造方法
@param carId ⻋型编号@param serialId ⻋ 系@return 实 例
/

  • (instancetype)initWithCarId:(NSInteger)carId serialId:(NSInteger)serialId;
    4).[强制] 代码实现中需要添加程序区块说明,#pragma mark - xxxx 注释。分区格式如下,区块位置保持统一,示例如下。说明:下方{}⻛格仅为节省空间,编码⻛格请参⻅【2.5 大括号 {}
    #pragma mark - Lifecycle
    -(instancetype)init {}
    -(void)dealloc {}
    -(void)viewDidLoad {}
    -(void)viewWillAppear:(BOOL)animated {}
    -(void)didReceiveMemoryWarning {} #pragma mark - Custom Accessors
    -(void)setCustomProperty:(id)value {}
    -(id)customProperty {} #pragma mark -IBActions
  • (IBAction)submitData:(id)sender {} #pragma mark -Public
    -(void)PublicMethod{}

#pragma mark -Private
-(void)PublicMethod{}
5).[强制] 优质代码一般是自注释的,若在程序中的,须对一些业务代码进行必要的说明,采用如下注释⻛格需要简要说明,能用一句话表述清楚,使用单行注释。
// 单行说明性注释,与上方空一行,与下方注释代码不空行[myObj somethingXXXXX];
对于一些重要的业务点,比较难理解的编码,需要详细说明,采用多行注释描述的方式。
/*
说明做了什么事
说明有哪些注意点?
说明有哪些要说明的问题?
与上方空一行,与下方注释代码不空行
/
[obj somethingXXXXX];
6).[建议] 对于未完善功能添加//TODO:标记,方便以后查找。
7).[建议] 对于确定无用或已废弃的代码,不应该以注释的形式存在于工程中,如果后面有可能恢复此段代码的要配合说明,而不是简单的注释掉。
8).[建议] 代码修改的同时,注释也要进行相应的修改,尤其是参数、返回值、异常、核心逻辑等的修改。
4.3延迟调用范围
1).[必须] performSelector:withObject:afterDelay:要在有Runloop 的线程里调用,否则调用无法生效。 说明:异步线
程默认是没有 runloop 的,除非手动创建;而主线程是系统会自动创建 Runloop 的。所以在异步线程调用是请先确保该线程是有 Runloop 的
2).使用performSelector:withObject:afterDelay:和 cancelPreviousPerformRequestsWithTarget 组合的时候要小心: afterDelay 会增加引用计数,而 cancel 会对引用计数减一 如果receiver 在引用计数器为 1 的时候,调用 cancel 会立即回收receiver。后续再次调用 receiver 的方法就会 crash。所以我们需要使用 weakSelf 并判空。如下:
weak typeof(self) weakSelf = self;
[NSObject cancelPreviousPerformRequestsWithTarget:self]; if (!weakSelf) {
// NSLog(@"self dealloc"); return;
}
[self doOther];
4.4类型设计范围
1).[建议] 尽量减少继承,类的继承关系不要超过 3 层。可以考虑使用category、protocol 来代替继承。
2).[建议] 把一些稳定的、公共的变量或者方法抽取到父类中。子类尽量只维持父类所不具备的特性和功能。
3).[建议] .h 文件中尽量不要声明成员变量。
4).[建议] .h 文件中的属性尽量声明为只读。
5).[建议] .h 文件中只暴露出一些必要的类、公开的方法、只读属性;私有类、私有方法和私有属性以及成员变量,尽量写在.m 文件中。

五.代码组织规范
5.1地址参考
1). [建议] 参考地址: raywenderlich/objective-c-style-guide
5.2ViewController 控制器代码规格
1). 代码规范如下:
#pragma mark - Lifecycle
-(instancetype)init {}
-(void)dealloc {}
-(void)viewDidLoad {}
-(void)viewWillAppear:(BOOL)animated {}
-(void)didReceiveMemoryWarning {} #pragma mark - Custom Accessors
-(void)setCustomProperty:(id)value {}
-(id)customProperty {} #pragma mark - IBActions
-(IBAction)submitData:(id)sender {} #pragma mark - Public
-(void)publicMethod {} #pragma mark - Private
-(void)privateMethod {}
#pragma mark - UITextFieldDelegate #pragma mark - UITableViewDataSource #pragma mark - UITableViewDelegate #pragma mark - NSCopying
-(id)copyWithZone:(NSZone )zone {} #pragma mark - NSObject
-(NSString
)description {}
[建议] 以上只是提供了组织代码的一种思路,如果有其他更好的组织方式,也不是不可以。

六.注意要点

  1. [强制]注意 nil 值不能传入 NSArray 和 NSDictionary 字面量,会导致崩解,传入前必须判断
    2).[强制] init 初始化方法与构造方法,应该循环 Apple 生成代码模块命名规则,返回类型应使用 instancetype 而不是
    id
    3).[强制] 私有属性与私有方法应该声明在类实现文件的延展 (匿名的类型) 中@interface NYTAdvertisement () @property(nonatomic,strong)GADBannerView googleAdView; @property(nonatomic,strong)ADBannerView iAdView; @property(nonatomic,strong)UIWebView *adXWebView;br/>@end
    4).[强制] 字符串可能会存在 nil 、null 等情况,页面展示需要进行过滤或容错

5).[强制] 使用线程安全模式实现单例,如下:

  • (instancetype)sharedInstance {
    static ClassName sharedInstance = nil;
    static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{ sharedInstance = [[self alloc] init];
    });
    return sharedInstance;
    }
    6).[建议] 使用框架文件,导入框架根的头文件,不建议分别导入
    说明:每一个框架都会有一个和框架同名的头文件,它包含了框架内接口的所有引用,在使用框架的时候, 应该直接引用这个根头文件,而不是其它子模块的头文件,即使是你只用到了其中的一小部分,编译器会自动完成优化的。
    7).[建议] 尽量使用 CGRect 相关方法来取值尽量使用:
    CGFloat width = CGRectGetWidth(frame); CGFloat height = CGRectGetHeight(frame); 尽量不用:
    CGFloat width = frame.size.width; CGFloat height = frame.size.height;
    8).[建议]不要使用 方法。尽管很多时候能用 new 代替 alloc init 方法,但这可能会导致调试内存时出现不可预料的问题。的规范就是使用 alloc init 方法。
    9).[建议]注意循环引用问题。在使用 block,deletgate,NSNotification,NSTimer 等的时候,注意要避免造成循环引用。10). [建议]不要在 viewWillAppear 中做耗时操作

七.工程项目结构规范
1).[强制] 总工程的项目结构Project
├── DB
├── General
│ ├── Classes
│ │ ├── TBKBaseViewController.h
│ │ ├── TBKBaseViewController.m
│ │ ├── TBKHorizontalView.h
│ │ └── TBKHorizontalView.m
│ ├── Views
│ │ ├── TPKPullToRefresh.h
│ │ ├── TPKPullToRefresh.m
│ │ ├── TPKScrollView.h
│ │ └── TPKScrollView.m
│ └── category
│ ├── UIImageView+DownLoader.h

│ ├── UIImageView+DownLoader.m
│ ├── UIViewController+Sizzle.h
│ └── UIViewController+Sizzle.m
├── Macro
│ ├── AppMacro.h
│ ├── NotificationMacro.h
│ ├── UitilsMacro.h
│ └── VendorMacro.h
├── Main
│ ├── AppDelegate.h
│ ├── AppDelegate.m
│ ├── SceneDelegate.h
│ └── SceneDelegate.m
├── Network
├── Other
│ ├── ViewController.h
│ └── ViewController.m
├── Resources
│ ├── DataFiles
│ ├── Images
│ └── Sounds
└── Sections
├── Controller
├── Model
└── View 2).Resources :资源目录
1.DataFiles :存放 XML 和.plist 文件 (存放数据本地文件) 2.Images :图片目录
3.Sounds :音频文件3).Macro :存放宏定义文件
├── Macro
│ ├── AppMacro.h
│ ├── NotificationMacro.h
│ ├── UitilsMacro.h
│ └── VendorMacro.h
...
4).Other :其他(用于界面测试类使用,其用完之后,则可以直接删除掉)
├── Other
│ ├── ViewController.h
│ └── ViewController.m
...

5).Network :服务器交互相关6).Main :程序入口处处理
├── Main
│ ├── AppDelegate.h
│ ├── AppDelegate.m
│ ├── SceneDelegate.h
│ └── SceneDelegate.m
...
7).DB :数据库操作目录(用于数据存储) 8).General :常用类目录
1.Classes :常用的工具类
2.Views :存放自定义的常用的 View 3.Category:类别
├── General
│ ├── Classes
│ │ ├── TBKBaseViewController.h
│ │ ├── TBKBaseViewController.m
│ │ ├── TBKHorizontalView.h
│ │ └── TBKHorizontalView.m
│ ├── Views
│ │ ├── TPKPullToRefresh.h
│ │ ├── TPKPullToRefresh.m
│ │ ├── TPKScrollView.h
│ │ └── TPKScrollView.m
│ └── Category
│ ├── UIImageView+DownLoader.h
│ ├── UIImageView+DownLoader.m
│ ├── UIViewController+Sizzle.h
│ └── UIViewController+Sizzle.m
...
9).Sections :项目功能部分,设计模式是MVC 设计模式(这里可以使用 MVVM、MVC)等设计模式1.Model :数据相关的 Model 文件
2.Controller :功能模块实现义务逻辑3.View :功能模块数据显示的View
└── Sections
├── Controller
├── Model
└── View
...