iOS代码规范

关于初始化方法《iOS代码规范:初始化方法》可以看这篇文章

头文件

头文件要写详细⼀一点注释,特别是⼯工具类,要详细说明其作⽤用,关联业务,注意 事项

导入需要使用到的其他class头文件,头文件按类型分类

// controller
#import “TextSelectionViewController.h"
// model
#import "PayPasswordUpdateModel.h"
#import "WalletModel.h"
// views
#import "PayPasswordAlertView.h"

类名

  1. 类的名称应该以三个大写字母为前缀
  2. 大写驼峰式命名。每个单词首字母大写。比如「申请记录控制器」ApplyRecordsViewController
  3. 每个类型的命名以该类型结尾。
    ViewController:使用 ViewController 结尾。例子:ApplyRecordsViewController
    View:使用 View 结尾。例子:分界线:boundaryView
    NSArray:使用 s 结尾。比如商品分类数据源。categories
    UITableViewCell:使用 Cell 结尾。比如 MyProfileCell
    Protocol:使用 Delegate 或者 Datasource 结尾。比如 XQScanViewDelegate
    Tool:工具类
    代理类:Delegate
    Service 类:Service
  4. 创建子类的时候,应该把代表子类特点的部分放在前缀和父类名的中间。
//父类
ZOCSalesListViewController

//子类
ZOCDaySalesListViewController
ZOCMonthSalesListViewController
  1. 在类的.h文件中尽量少引用其他头文件
    类A需要将类B的实例变量作为它公共API的属性。这个时候,我们不应该引入类B的头文件,而应该使用向前声明(forward declaring)使用class关键字,并且在A的实现文件引用B的头文件。
// EOCPerson.h
#import <Foundation/Foundation.h>

@class EOCEmployer;

@interface EOCPerson : NSObject

@property (nonatomic, copy) NSString *firstName;
@property (nonatomic, copy) NSString *lastName;
@property (nonatomic, strong) EOCEmployer *employer;//将EOCEmployer作为属性

@end

// EOCPerson.m
#import "EOCEmployer.h"

优点:不在A的头文件中引入B的头文件,就不会一并引入B的全部内容,这样就减少了编译时间。

  1. 在类声明中包含多个protocal,每个protocal占用一行
@interface ShopViewController () <UIGestureRecognizerDelegate,
                                  HXSClickEventDelegate,
                                  UITableViewDelegate,
                                  UITableViewDataSource>

方法

  1. 方法与方法之间间隔一行
  2. 方法最后面的括号需要另起一行。遵循 Apple 的规范;对于其他场景的括号,括号不需要单独换行。比如 if 后面的括号
  3. 如果方法参数过多过长,建议多行书写。用冒号进行对齐
- (instancetype)init
{
    self = [super init];
    if (self) {
        <#statements#>
    }
    return self;
}

- (void)doHomework:(NSString *)name
            period:(NSInteger)second
            score:(NSInteger)score;
  1. 一个方法内的代码最好保持在50行以内,一般经验来看如果一个方法里面的代码行数过多,代码的阅读体验就很差(别问为什么,做过重构代码行数很长的人都有类似的心情)
  2. 一个函数只做一个事情,做到单一原则。所有的类、方法设计好后就可以类似搭积木一样实现一个系统。
  3. 对于有返回值的函数,且函数内有分支情况。确保每个分支都有返回值
  4. 函数如果有多个参数,外部传入的参数需要检验参数的非空、数据类型的合法性,参数错误做一些措施:立即返回、断言
  5. 多个函数如果有逻辑重复的代码,建议将重复的部分抽取出来,成为独立的函数进行调用
  6. 方法如果有多个参数的情况下需要注意是否需要介词和连词。很多时候在不知道如何抉择测时候思考下苹果的一些 API 的方法命名
//good
- (instancetype)initWithAge:(NSInteger)age name:(NSString *)name;

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath;


//bad
- (instancetype)initWithAge:(NSInteger)age andName:(NSString *)name;

- (void)tableView:(UITableView *)tableView :(NSIndexPath *)indexPath;

方法调用沿用声明方法的习惯

[self doSomethingWith:@"test" rect:self.view.frame interval:1.0f];

[self doSomethingWith:@"test"
                 rect:self.view.frame
             interval:1.0f];

相关的赋值语句等号对齐

promotionsEntity.promotionImageStr   = activityItemDict[@"promotion_image"];
promotionsEntity.promotionIdNum      = activityItemDict[@"promotion_id"];

类中功能模块以#pragma mark – 分割,上空两行,下空一行

#pragma mark – UITextFieldDelegate
  1. 控制器器的方法分类及顺序规范:
#pragma mark - Initialization

#pragma mark - Life cycle

#pragma mark - Actions
// view的⼀一些事件⽐比如按钮点击、⼿手势事件处理理、通知处理理

#pragma mark - Delegates // 实现遵循的代理理⽅方法

#pragma mark - Private method // 控制器器的⼀一些私有的辅助⽅方法

#pragma mark - Getter & Setter // 属性的Getter/Setter⽅方法
  1. 除了 .m 文件中方法,其他的地方大括号"{"不需要另起一行。推荐:
if (!error) {
    return success;
}

- (void)doHomework
{
    if (self.hungry) {
        return;
    }
    //doSomething
}

属性

  • 对外尽量使用不可变对象
    尽量把对外公布出来的属性设置为只读,在实现文件内部设为读写。具体做法是:
    在头文件中,设置对象属性为readonly。在实现文件中设置为readwrite。

这样一来,在外部就只能读取该数据,而不能修改它,使得这个类的实例所持有的数据更加安全。而且,对于集合类的对象,更应该仔细考虑是否可以将其设为可变的。
如果在公开部分只能设置其为只读属性,那么就在非公开部分存储一个可变型。所以当在外部获取这个属性时,获取的只是内部可变型的一个不可变版本,例如:

@interface EOCPerson : NSObject

@property (nonatomic, copy, readonly) NSString *firstName;
@property (nonatomic, copy, readonly) NSString *lastName;
@property (nonatomic, strong, readonly) NSSet *friends //向外公开的不可变集合

- (id)initWithFirstName:(NSString*)firstName andLastName:(NSString*)lastName;
- (void)addFriend:(EOCPerson*)person;
- (void)removeFriend:(EOCPerson*)person;

@end

在这里,我们将friends属性设置为不可变的set。然后,提供了来增加和删除这个set里的元素的公共接口。

@interface EOCPerson ()

@property (nonatomic, copy, readwrite) NSString *firstName;
@property (nonatomic, copy, readwrite) NSString *lastName;

@end

@implementation EOCPerson {
     NSMutableSet *_internalFriends;  //实现文件里的可变集合
}

- (NSSet*)friends 
{
     return [_internalFriends copy]; //get方法返回的永远是可变set的不可变型
}

- (void)addFriend:(EOCPerson*)person 
{
    [_internalFriends addObject:person]; //在外部增加集合元素的操作
    //do something when add element
}

- (void)removeFriend:(EOCPerson*)person 
{
    [_internalFriends removeObject:person]; //在外部移除元素的操作
    //do something when remove element
}

- (id)initWithFirstName:(NSString*)firstName andLastName:(NSString*)lastName 
{

     if ((self = [super init])) {
        _firstName = firstName;
        _lastName = lastName;
        _internalFriends = [NSMutableSet new];
    }
 return self;
}

这里最重要的代码是:

- (NSSet*)friends 
{
   return [_internalFriends copy];
}

这个是friends属性的获取方法:它将当前保存的可变set复制了一不可变的set并返回。因此,外部读取到的set都将是不可变的版本。

  • 属性应该尽可能描述性地命名,避免缩写,并且是小写字母开头的驼峰命名
例:NSString *text;   不要这样 : NSString* text;NSString * text;
@property (nonatomic, strong) UITableView *tableView;
@property (nonatomic, assign, readonly) BOOL loading;   
@property (nonatomic, weak) id<#delegate#> delegate;
@property (nonatomic, copy) <#returnType#> (^<#Block#>)(<#parType#>);

条件表达式

  1. 当有条件过多、过长的时候需要换行,为了代码看起来整齐些
//good
if (condition1() && 
    condition2() && 
    condition3() && 
    condition4()) {
  // Do something
}
//bad
if (condition1() && condition2() && condition3() && condition4()) { // Do something }
  1. 在一个代码块里面有个可能的情况时善于使用 return 来结束异常的情况
- (void)doHomework
{
    if (self.hungry) {
        return;
    }
    if (self.thirsty) {
        return;
    }
    if (self.tired) {
        return;
    }
}
  1. 每个分支的实现都必须使用 {} 包含。
// bad
if (self.hungry) self.eat() 
// good
if (self.hungry) {
    self.eat()
}
  1. 条件判断的时候应该是变量在左,条件在右。 if ( currentCursor == 2 ) { //... }
  2. switch 语句后面的每个分支都需要用大括号括起来
  3. switch 语句后面的 default 分支必须存在,除非是在对枚举进行 switch
switch (menuType) {  
  case menuTypeLeft: {
    // ...  
    break; 
   }
  case menuTypeRight: {
    // ...  
    break; 
  }
  case menuTypeTop: {
    // ...  
    break; 
  }
  case menuTypeBottom: {
    // ...  
    break; 
  }
}

变量命名

  • 做到⻅见名知意
  • 一个变量有且只有一个功能,尽量不要把一个变量用作多种用途
  • 变量的名称必须同时包含功能与类型
UIButton *addBtn 
UILabel *nameLbl 
NSString *addressStr
  • 系统常用类作实例变量声明时加入后缀


    系统常用类作实例变量声明时加入后缀.png

注释

YES upon success; NO upon failure ---- 成功返回YES 失败返回NO

  1. 类的注释写在当前类文件的顶部
  2. 对于属性的注释建议写在属性上面,用的时候,会有提示功能
/// 刷新按钮
@property (nonatomic, strong) UIButton *refreshBtn;
  1. 对于 .h 文件中方法的注释,一律按快捷键 command+option+/。三个快捷键解决。按需在旁边对方法进行说明解释、返回值、参数的说明和解释
    对于.m文件中方法的注释,在方法的上边或右边添加//,注释符和注释内容需要间隔一个空格。
// load network data
  1. 功能注释
    版本迭代中,在同事写的代码基础上开发,一定要写上版本功能注释,方便询问具体功能
// 这是一个新加的功能 v5.20.0 by minjing.lin  2019-03-20 18:00

宏定义

  • 尽量量少⽤用宏来定义具体类型的数据,可以改⽤用static NSString* const kMsg = @”MSG”等来表示;
  • 字⺟母全⼤大写,单词与单词之间用_分割
  • 采⽤用驼峰法命名(首字母大写)
#define HOME_PAGE_DID_SCROLL @"com.xq.home.page.tableview.did.scroll"
#define KHomePageDidScroll @"com.xq.home.page.tableview.did.scroll"

图片资源命名

  • 单个文件的命名

文件资源的命名也需要一定的规范
命名规范:不能有中文、大写、特殊符号、空白
形式为:功能模块名类别功能状态@nx.png
Setting_Button_search_selected@2x.png
Setting_Button_search_unselected@2x.png
万能公式:类别
功能模块页面名称状态.png
icon_tab_bookshelf_sel@2x.png

  • 资源的文件夹命名
    最好也参考 App 按照功能模块建立对应的实体文件夹目录,最后到对应的目录下添加相应的资源文件。

版本规范

采用 A.B.C 三位数字命名,比如:1.0.2,当有更新的情况下按照下面的依据


版本规范.png

注意事项

  • 使⽤用block时,如果block被单例例持有,则block内的self应使⽤用弱引⽤用
  • 函数(方法)块之间使用一个空行分隔
  • 对输入参数的正确性和有效性进行检查
Xcode 代码块的存放地址
~/Library/Developer/Xcode/UserData/CodeSnippets 
Xcode文件模版的存放地址
/Applications/Xcode.app/Contents/Developer/Platforms/iPhoneOS.platform/Developer/Library/Xcode/Templates/File Templates/
一元运算符与变量之间没有空格:
!aValue
-aValue  //负号
~aValue  //位非
++iCount
*strSource
169910077b339250.jpeg

参考1
参考2
规范

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

推荐阅读更多精彩内容

  • 概要 Objective-C是一门面向对象的动态编程语言,主要用于编写iOS和Mac应用程序。关于Objectiv...
    DreamMmMmM阅读 1,160评论 0 7
  • 一些原则 参照 https://github.com/FantasticLBP/codesnippets 长的,描...
    这小子阅读 695评论 0 0
  • 一、命名规范 命名规则对于维护代码来说是非常重要的,。Objective-C方法名往往很长,不过这也有好处,让很多...
    jmsYang阅读 332评论 0 0
  • iOS代码规范总结 一些原则 长的,描述性的方法和变量命名是好的。不要使用简写,除非是一些大家都知道的场景比如 V...
    ElegantLiar阅读 1,597评论 0 17
  • iOS代码规范 一、前言 本规范基于Google Objective-C Style Guide和百度Object...
    CoderHw阅读 896评论 0 7