Sivan Wu's blog

A developer, a learner

View on GitHub

iOS编码规范

官方的编码规范文档 Naming Basics

1. 命名规则

1.1 通用命名规则

通常指变量,常量,属性,参数,函数等的的命名规范General Principles

【要】自我描述:用英文描述自身的意义,杜绝拼音,过度缩写或者无意义的命名

【要】驼峰命名规则,有些特殊的不需要驼峰,例如:FTP、WWW

【不要】自我指涉:末尾加上自己类型的后缀;通知、掩码的命名除外

命名 说明
person 这是NSObject类型对象,这样可以
personObject NSObject类型对象,这样不合法
NSUnderlineByWordMask 掩码这样写合法
NSTableViewColumnDidMoveNotification 通知这样写也合法

1.2 缩写规范

通常不应该缩写单词,除非是一些总所周知的缩写

1.3 方法命名规范

- (void)invokeWithTarget:(id)target;
- (void)selectTabViewItem:(NSTabViewItem *)tabViewItem

1.4 Accessor命名规范

属性的set和get方法 General Rules

名词

- (NSString *)title;
- (void)setTitle:(NSString *)aTitle;

形容词

- (BOOL)isEditable;
- (void)setEditable:(BOOL)flag;

动词

- (BOOL)showsAlpha;
- (void)setShowsAlpha:(BOOL)flag;

1.5 delegate命名规范

- (BOOL)tableView:(NSTableView *)tableView shouldSelectRow:(int)row;
- (BOOL)application:(NSApplication *)sender openFile:(NSString *)filename;
- (BOOL)applicationOpenUntitledFile:(NSApplication *)sender;
- (void)windowDidChangeScreen:(NSNotification *)notification;
- (void)browserDidScroll:(NSBrowser *)sender;
- (NSUndoManager *)windowWillReturnUndoManager:(NSWindow *)window;

1.6 Private方法命名规范

大部分情况下,私有方法的命名和公有方法的命名规则是一样的。然而,通常情况下应该给私有方法添加一个前缀,目的是和公有方法区分开。尽管这样,这种给私有方法加前缀的命名方式有可能引起一些奇怪的问题。问题就是:当你从Cocoa framework(即Cocoa系统库)中的某个类派生出来一个子类时,你并不知道你的子类中定义的私有方法是否覆盖了父类的私有方法,即有可能你自己在子类中实现的私有方法和父类中的某个私有方法同名。在运行时,这极有可能导致一些莫名其妙的问题,并且调试追踪问题的难度也是相当大。

Cocoa frameworks(Cocoa系统库)中的私有方法通常以一个下划线“ _ ”开头,用于标记这些方法是私有的(比如, _fooData ) ,这大概就是Apple工程师的开发习惯。基于这个事实,提供以下两条建议

1. 禁止使用下划线“ _ “作为私有方法的开头。Apple已经预留这种私有方法的命名习惯

2. 如果你是要子类化Cocoa Frameworks中的一个非常庞大复杂的类(比如NSView或UIView),并且你想绝对的确保你自己的子类中的私有方法名和父类中的私有方法名不重复。你可以添加一个你自己的前缀作为私有方法的前缀,这个前缀应该尽可能的独特。也许这个前缀是基于你公司或者项目的缩写,比如”XX_“。尽管给私有方法增加前缀看起来和”方法存在于他们的类的命名空间中“这一之前的说法有些冲突,但此处的意图是:为子类私有方法添加前缀仅仅是为了保证子类方法和父类方法名称不冲突。

1.7 Catogory命名规范

1. category中不要声明属性和成员变量,尽管可以使用runtime方法来实现,但是这样会导致不方便管理

2. 避免category中的方法覆盖系统方法。可以使用前缀来区分系统方法和category方法。但前缀不要仅仅使用下划线”_“

3. 如果一个类比较复杂,建议使用category的方式组织代码

1.8 Class命名规范

class的名称应该由两部分组成,前缀+名称。即,class的名称应该包含一个前缀和一个名词

1.9 Protocol命名规范

如果proctocol不仅声明了一堆相关方法,还关联了某个class。这种关联class的protocol的命名取决于关联的class,然后再后面再加上protocol或delegate用于显示的声明这是一份协议

命名 说明
UITableViewDelegate OK
NSObjectProtocol OK

1.10 Notification命名规范

1. 规则:[Name of associated class] + [Did | Will] + [UniquePartOfName] + Notification

NSApplicationDidBecomeActiveNotification
NSWindowDidMiniaturizeNotification
NSTextViewDidChangeSelectionNotification
NSColorPanelColorDidChangeNotification

2. object通常是指发出notification的对象,如果在发送notification的同时要传递一些额外的信息,请使用userInfo,而不是object

3. 如果某个通知是为了告知外界某个事件”即将”发生或者”已经”发生,则请在通知名称中使用“will”或者“did”这样的助动词,见上

1.11 Constant命名规范

可以使用k开头为constant命名,Constants规范

不要使用预处理指令创建常量

1.12 Exception命名规范

类似于Notification命名规范 [Prefix] + [UniquePartOfName] + Exception

NSColorListIOException
NSColorListNotEditableException
NSDraggingException
NSFontUnavailableException
NSIllegalSelectorException

2. 编码规则

2.1 Initialize规范

Tips and Techniques for Framework Developers

initialize方法的调用遵循继承规则(所谓继承规则,简单来讲是指:子类方法中可以调用到父类的同名方法,即使没有显式调用[super xxx])。如果我们没有实现initialize方法,运行时初次调用这个类的时候,系统会沿着继承链,先后给继承链上游中的每个超类发送一条initialize消息,直到某个超类实现了initlialize方法,才会停止向上调用。因此,在运行时,某个类的initialize方法可能会被调用多次(比如:一个子类没有实现initialize方法)。 比如:有三个类:SuperClass、SubClass和FinalClass。他们的继承关系是这样的FinalClass->SubClass->SuperClass,只实现了SuperClass方法的initialize方法。

// SuperClass
@implementationSuperClass
+ (void)initialize {
    NSLog(@"superClass initalize");
}
@end
// 初始化FinalClass
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    FinalClass *finalC = [FinalClass new];
}
// 控制台输出结果
2018-01-2722:11:03.130365+0800 Demo[67162:11721965] superClass initalize
2018-01-2722:11:03.130722+0800 Demo[67162:11721965] superClass initalize
2018-01-2722:11:03.130815+0800 Demo[67162:11721965] superClass initalize

解释

因为FinalClass继承自SubClass,SubClass继承自SuperClass。因为继承体系中只有SuperClass实现了initialize方法,导致初始化FinalClass这个子类时,FinalClass会调用他的父类(SubClass)中的initialize方法。又因为他(FinalClass)的父类(SubClass)也没有实现initialize方法,又会继续沿着继承体系,向上游寻找,最后找到SubClass的父类(SuperClass)。因为SuperClass实现了这个initialize方法,所以调用结束。至于为什么是连续调用了三次SuperClass的initialize方法。因为子类FinalClass的初始化触发了超类SubClass、SuperClass的初始化。所以初始化FinalClass时,实际上使这三个类都得到了初始化的机会,自然就会连续调用三次SuperClass的initialize方法

还是上面那三个类,如果我们又给SubClass实现了initialize方法,那么控制台将会输出如下结果:

2018-01-27 22:34:54.697952+0800Load[67652:11780578]superClassinitalize
2018-01-27 22:34:54.698118+0800Load[67652:11780578]subClassinitialize
2018-01-27 22:34:54.698472+0800Load[67652:11780578]subClassinitialize
+ (void)initialize {
    staticdispatch_once_t onceToken = 0;
    dispatch_once(&onceToken, ^{
        // the initializing code
    }
}

【建议】如果我们想在继承体系的某个指定的类的initialize方法中执行一些初始化代码,可以使用类型检查和而非dispatch_once()。如下

if (self == [NSFooclass]) {
    // the initializing code
}

说了这么多,总而言之,由于任何子类都会调用父类的initialize方法,所以可能会导致某个父类的initialize方法会被调用多次,为了避免这种情况,我们可以使用

  1. 类型判断
  2. dispatch_once()

这两种方式,以保证initialize中的代码不会被无故调用。

2.2 初始化方法规范

Objective-C有designated Initializers和secondary Initializers的概念。

一个类可以有一个或者多个designated Initializers。但是要保证所有的其他secondary initializers都要调用designated Initializers。即:只有designated Initializers才会存储对象的信息。这样的好处是:当这个类底层的某些数据存储机制发生变化时(可能是一些property的变更),只需要修改这个designated Initializers内部的代码即可。无需改动其他secondary Initializers初始化方法的代码。

1. 所有secondary 初始化方法都应该调用designated 初始化方法
2. 所有子类的designated初始化方法都要调用父类的designated初始化方法。使这种调用关系沿着类的继承体系形成一条链
3. 如果子类的designated初始化方法与超类的designated初始化方法不同,则子类应该覆写超类的designated初始化方法。(因为开发者很有可能直接调用超类的某个designated方法来初始化一个子类对象,这样也是合情合理的,但使用超类的方法初始化子类,可能会导致子类在初始化时缺失一些必要信息)
4. 如果超类的某个初始化方法不适用于子类,则子类应该覆写这个超类的方法,并在其中抛出异常
5. 禁止子类的designated初始化方法调用父类的secondary初始化方法。否则容易陷入方法调用死循环。如下
// 超类
@interfaceParentObject : NSObject
@end
@implementationParentObject
    //designated initializer    
    - (instancetype)initWithURL:(NSString*)url title:(NSString*)title {
        if (self = [super init]) {
            _url = [url copy];
            _title = [title copy];
        }
        returnself;
    }
    //secondary initializer
    - (instancetype)initWithURL:(NSString*)url {
        return [self initWithURL:url title:nil];
    }
    @end
// 子类
@interfaceChildObject : ParentObject
@end
    @implementationChildObject
    //designated initializer
    - (instancetype)initWithURL:(NSString*)url title:(NSString*)title {
        //在designated intializer中调用 secondary initializer,错误的
        if (self = [super initWithURL:url]) {
        }
        returnself;
    }
@end
@implementationViewController
    - (void)viewDidLoad {
        [super viewDidLoad];
        // 这里会死循环
        ChildObject* child = [[ChildObject alloc] initWithURL:@"url" title:@"title"];
    }
@end
6. 另外禁止在init方法中使用self.xxx的方式访问属性。如果存在继承的情况下,很有可能导致崩溃。为什么不能在init和dealloc函数中使用accessor方法

2.3 初始化错误

  1. 调用父类的designated初始化方法初始化本类的对象
  2. 校验父类designated初始化方法返回的对象是否为nil
  3. 如果初始化当前对象的时候发生了错误,应该给予对应的处理:释放对象,并返回nil。

以下实例列举类初始化阶段可能会存在的错误:

- (id)init {
    self = [super init];  // Call a designated initializer here.
    if (self != nil) {
        // Initialize object  ...
        if (someError) {
            [self release];
            self = nil;
        }
    }
    returnself;
}

2.4 dealloc规范

1. 不要忘记在dealloc方法中移除通知和KVO
2. dealloc 方法应该放在实现文件的最上面,并且刚好在 @synthesize 和 @dynamic 语句的后面。在任何类中,init 都应该直接放在 dealloc 方法的下面
3. 在dealloc方法中,禁止将self作为参数传递出去,如果self被retain住,到下个runloop周期再释放,则会造成多次释放crash
-(void)dealloc{
    [self unsafeMethod:self];
    // 因为当前已经在self这个指针所指向的对象的销毁阶段,销毁self所指向的对象已经在所难免。
    // 如果在unsafeMethod:中把self放到了autorelease poll中,那么self会被retain住,计划下个runloop周期在进行销毁。
    // 但是dealloc运行结束后,self所指向的对象的内存空间就直接被回收了,但是self这个指针还没有销毁(即没有被置为nil),导致self变成了一个名副其实的野指针。
    // 到了下一个runloop周期,因为self所指向的对象已经被销毁,会因为非法访问而造成crash问题。
}
4. 和init方法一样,禁止在dealloc方法中使用self.xxx的方式访问属性。如果存在继承的情况下,很有可能导致崩溃。

2.5 block规范

  1. 调用block时,要记得判空
  2. 注意block可能的循环引用

2.6 Notification使用规范

通知和block,delegate所起到的作用是相同,无非是在类之间传递信息 当夸域较远时,可以很好地用通知解决信息传递 但是当通知过多的时候,会容易产生bug,可读性和维护性也差,不必要的耦合会过高

  1. 基于以上的陈述,当我们使用通知时,必须要思考,有没有更好的办法来代替这个通知。禁止遇到问题就想到通知,把通知作为备选项而非首选项
  2. post通知时,object通常是指发出notification的对象,如果在发送notification的同时要传递一些额外的信息,请使用userInfo,而不是object
  3. 通知执行的线程取决于发出通知的线程,而不是注册通知的线程。所以需要在一些UI通知方法内,回到主线程,因为调用者可能不知道发出的地方是不是主线程

说明:每个进程都会创建一个NotificationCenter,这个center通过NSNotificationCenter defaultCenter获取,当然也可以自己创建一个center。 NoticiationCenter是以同步(非异步,当前线程,会等待,会阻塞)的方式发送请求。即,当post通知时,center会一直等待所有的observer都收到并且处理了通知才会返回到poster。如果需要异步发送通知,请使用notificationQueue,在一个多线程的应用中,通知会发送到所有的线程中

2.7 UI规范

  1. 如果想要获取window,不要使用view.window获取。请使用[[UIApplication sharedApplication] keyWindow]
  2. 在使用到 UIScrollView,UITableView,UICollectionView 的 Class 中,需要在 dealloc 方法里手动的把对应的 delegate, dataSouce 置为 nil
  3. UITableView使用self-sizing实现不等高cell时,请在- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath; 中给cell设置数据。不要在- (void)tableView:(UITableView *)tableView willDisplayCell:(UITableViewCell *)cell forRowAtIndexPath:(NSIndexPath *)indexPath;方法中给cell设置数据
  4. 当访问一个 CGRect 的 x, y, width, height 时,应该使用CGGeometry 函数代替直接访问结构体成员。苹果的 CGGeometry 参考中说到

All functions described in this reference that take CGRect data structures as inputs implicitly standardize those rectangles before calculating their results. For this reason, your applications should avoid directly reading and writing the data stored in the CGRect data structure. Instead, use the functions described here to manipulate rectangles and to retrieve their characteristics.

推荐写法

CGRect frame = self.view.frame;

CGFloat x = CGRectGetMinX(frame);
CGFloat y = CGRectGetMinY(frame);
CGFloat width = CGRectGetWidth(frame);
CGFloat height = CGRectGetHeight(frame);

不推荐写法

CGRect frame = self.view.frame;

CGFloat x = frame.origin.x;
CGFloat y = frame.origin.y;
CGFloat width = frame.size.width;
CGFloat height = frame.size.height;

2.8 IO规范

尽量少用NSUserDefaults。

说明:[[NSUserDefaults standardUserDefaults] synchronize] 会block住当前线程,知道所有的内容都写进磁盘,如果内容过多,重复调用的话会严重影响性能

一些经常被使用的文件建议做好缓存。避免重复的IO操作。建议只有在合适的时候再进行持久化操作

2.9 Collection规范

1. 不要用一个可能为nil的对象初始化集合对象,否则可能会导致crash,对插入到集合对象里面的对象也要进行判空
// 可能崩溃
NSObject *obj = somOjbcetMaybeNil;
NSMutableArray *arrM = [NSMutableArray arrayWithObject:obj];
// 崩溃信息:
*** Terminating app due to uncaught exception'NSInvalidArgumentException', 
reason: '*** -[__NSPlaceholderArray initWithObjects:count:]: 
attempt to insert nil object from objects[0]’

改进

// 改进办法:
NSObject *obj = somOjbcetMaybeNil;
NSMutableArray *arrM = nil;
if (obj && [obj isKindOfClass:[NSObjectclass]]) {
    arrM = [NSMutableArray arrayWithObject:obj];
} else {
    arrM = nil;
}
2. 注意在多线程环境下访问可变集合对象的问题,必要时应该加锁保护。不可变集合(比如NSArray)类默认是线程安全的,而可变集合类(比如NSMutableArray)不是线程安全的
3. 遍历可变集合对象中的元素的时候,应该先copy为不可变集合,再进行遍历
// 正确写法
- (void)checkAllValidItems{
NSArray *array = [array copy];
[array enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
//do something using obj
}]; }

// 错误写法
- (void)checkAllValidItems{
[self.allItems enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
    //do something using obj
    // 如果在enumerate过程中,其他线程对allItems这个可变集合进行了变更操作,这里就有可能引发crash
}]; }
4. 注意使用enumerateObjectsUsingBlock遍历集合对象中的对象时,关键字return的作用域。block中的return代表的是使当前的block返回,而非使当前的整个函数体返回。以下使用NSArray举例,其他集合类型同理
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    NSArray *array = [NSArray arrayWithObject:@"1"];
    [array enumerateObjectsUsingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
        // excute some code...
        return;
    }];
    // 依然会执行到这里
    NSLog(@"fall through");
}

// 执行结果:
// fall through

当然,两个enumerateObjectsUsingBlock嵌套,如果仅在最内层的block中return,外层block的代码还是会被执行。如下:

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    NSArray *arr1 = [NSArray arrayWithObject:@"1"];
    NSArray *arr2 = [NSArray arrayWithObject:@"2"];
    [arr2 enumerateObjectsUsingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
        [arr1 enumerateObjectsUsingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
            // do something
            return;
        }];
        NSLog(@"fall through");
    }];
    NSLog(@"fall through");
}

// 执行结果:
// fall through
// fall through

这么理解,block是一个匿名函数,block内的返回,是函数的返回。

5. 禁止返回mutable对象,禁止mutable对象作为入参传递,这是避免不可预知crash的良药,再代码量大了以后,很可能会有多线程导致闪退的问题
6. 建议集合类使用泛型来指定对象的类型
@property(nonatomic,copy) NSArray<NSString *> *array;
@property(nonatomic,strong) NSMutableDictionary<NSString *,NSString *> *dictionary;

2.10 分支语句规范

1. if条件判断语句后面必须要加大括号{}。不然随着业务的发展和代码迭代,极有可能引起逻辑问题,否则随着代码量的增大,会导致可读性差,理解逻辑出错
// 建议
if (!error) {
    return success;
}
// 不建议
if (!error) 
    return success;
if (!error)  return success;
2. 遵循gold path法则,不要把真正的逻辑写道括号内
// 不建议
- (void)someFuncWith:(NSString *)parameter {
    if (parameter) {
        // do something
        [self doSomething];
    }
}
// 建议
- (void)someFuncWith:(NSString *)parameter {
    if (!parameter) {
        return;
    }
    // do something
    [self doSomething];
}
3. 对于条件语句的真假,因为 nil 解析为 NO,所以没有必要在条件中与它进行比较。永远不要直接和 YES 和 NO进行比较,因为 YES 被定义为 1,而 BOOL 可以多达 8 位
// 建议
if (isAwesome)
if (![someObject boolValue])
// 禁止这样做
if ([someObject boolValue] == NO) { }
if (isAwesome == YES) { } 
4. switch…case…语句的每个case都要添加break关键字,避免出现fall-through

2.11 对象等同性判断

isEqual:方法允许我们传入任意类型的对象作为参数,如果参数类型和receiver(方法调用者)类型不一致,会返回NO。

而isEqualToString:和isEqualToArray:这两个方法会假设参数类型和receiver类型一致,也就是说,这两个方法不会对参数进行类型检查,因此这两个方法性能更好但不安全。

如果我们是从外部数据源(比如info.plist或preferences)获取的数据,那么推荐使用isEqual:,因为这样更安全。如果我们知道参数的确切类型,那么可以使用类似于isEqualToString:这样的方法,因为性能更好。

如果有需要,可以重载等同性判断方法,对象等同性判断

2.12 懒加载规范

适用场景

如果一个对象在懒加载后,某些场景下又被置为nil,很难保证这个对象的懒加载不会再次被触发

2.13 多线程规范

  1. 禁止使用GCD的dispatch_get_current_queue(),会获得意想不到的信息,已经被苹果废弃
  2. 仅当必须保证顺序执行时才使用dispatch_sync,否则容易出现死锁,应避免使用,可使用dispatch_async
  3. 禁止在非主线程中进行UI元素的操作
  4. 如果需要进行大文件或者多文件的IO操作,禁止主线程使用,必须进行异步处理
  5. 对剪贴板的读取必须要放在异步线程处理,最新Mac和iOS里的剪贴板共享功能会导致有可能需要读取大量的内容,导致读取线程被长时间阻塞
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
   UIPasteboard *pasteboard = [UIPasteboard generalPasteboard]; 
   if (pasteboard.string.length > 0) {//这个方法会阻塞线程
      NSString *text = [pasteboard.string copy];
      [pasteboard setValue:@"" forPasteboardType:UIPasteboardNameGeneral];
      if (text == nil || [text isEqualToString:@""]) {
          return ;
      }
      dispatch_async(dispatch_get_main_queue(), ^{
          [self processShareCode:text];
      });
   }
});

2.14 内存管理规范

  1. 函数提前return的时候,要注意是否有对象没有被释放掉(CF对象),否则会内存泄漏
  2. 少用单例,会造成常驻内存。UIApplication、access database 、request network 、access userInfo这类全局仅存在一份的对象或者需要多线程访问的对象,可以使用单例
  3. 单例初始化方法中尽量保证单一职责,尤其不要进行其他单例的调用。
  4. 在dealloc方法中,禁止将self作为参数传递出去,如果self被retain住,到下个runloop周期再释放,则会造成多次释放crash
  5. 除非你清除的知道自己在做什么。否则不建议将UIView类的对象加入到NSArray、NSDictionary、NSSet中。如有需要可以添加到NSMapTable 和 NSHashTable。因为NSArray、NSDictionary、NSSet会对加入的对象做strong引用(即使你把加入的对象进行了weak)。而NSMapTable、NSHashTable会对加入的对象做weak引用。

说明:简单的说,NSHashTable相当于weak的NSMutableArray;NSMapTable相当于weak的NSMutableDictionary

// 错误的例子:
@implementationWSObject
- (void)dealloc {
    NSLog(@"dealloc");
}
@end
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    WSObject *object = [WSObject new];
    // 即使对object进行了weak弱化,数组也会强引用这个object对象。dealloc方法不会被执行。
    __weak typeof(object) weakObject = object;
    [self.arrM addObject:weakObject];
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        NSLog(@"count = %ld",self.arrM.count);
    });
}
// 打印结果:
// count = 1
// 正确的例子:
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    WSObject *object = [WSObject new];
    NSHashTable *hashTable = [NSHashTable weakObjectsHashTable];
    [hashTable addObject:object];
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        NSLog(@"count = %ld",hashTable.count);
    });
}
// 打印结果:
// dealloc
// count = 1

你可能对上面的例子有所疑惑,object已经释放了,但是控制台仍然输出 hashTable.count == 1。但是请相信我,此时存在于hashTable中的那个object已经变成了nil。不信你继续看下面的例子:

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    WSObject *object = [WSObject new];
    NSHashTable *hashTable = [NSHashTable weakObjectsHashTable];
    [hashTable addObject:object];
    dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
        NSLog(@"count = %ld",hashTable.count);
        if (hashTable && hashTable.count) {
            WSObject *object = [hashTable anyObject];
            NSLog(@"object = %@",[object self]);
        }
    });
}
// 打印结果:
2017-07-0422:19:10.952139+0800 tst[46834:4305636] dealloc
2017-07-0422:19:13.149903+0800 tst[46834:4305636] count = 1
2017-07-0422:20:55.234522+0800 tst[46834:4305636] object = (null)

2.15 延迟调用规范

performSelector:withObject:afterDelay:要在有Runloop的线程里调用,否则调用无法生效

异步线程默认是没有runloop的,除非手动创建;而主线程是系统会自动创建Runloop的。所以在异步线程调用是请先确保该线程是有Runloop的

使用performSelector:withObject:afterDelay:和cancelPreviousPerformRequestsWithTarget组合的时候要小心:

  1. afterDelay会增加引用计数,而cancel会对引用计数减一

  2. 如果receiver在引用计数器为1的时候,调用cancel会立即回收receiver。后续再次调用receiver的方法就会crash。所以我们需要使用weakSelf并判空。如下:

__weak typeof(self) weakSelf = self;
[NSObject cancelPreviousPerformRequestsWithTarget:self]; 
if (!weakSelf) {
    // NSLog(@"self dealloc");
    return;
 }
[self doOther];

2.16 注释规范

  1. 如果方法、函数、类、属性等需要提供给外界或者他人使用,必须要加注释说明
  2. 如果你的代码以SDK的形式提供给其他人使用,那么接口的注释是必须的。必须对暴露给外界的所有方法、属性、参数加以注释说明
  3. 注释应该说明其作用以及注意事项
  4. 因为方法或属性本身就具有自我描述性,注释应该简明扼要,说明是什么和为什么即可

2.17 类设计规范

  1. 尽量减少继承,类的继承关系不要超过3层,可以考虑使用category、protocol来代替继承
  2. 把一些稳定的、公共的变量或者方法抽取到父类中。子类尽量只维持父类所不具备的特性和功能
  3. .h文件中尽量不要声明成员变量
  4. .h文件中的属性尽量声明为只读
  5. .h文件中只暴露出一些必要的类、公开的方法、只读属性;私有类、私有方法和私有属性以及成员变量,尽量写在.m文件中

2.18 代码组织规范

objective-c-style-guide

#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 {}

2.19 工程结构规范

  1. 为了避免文件杂乱,磁盘文件应该保持和 Xcode 项目文件同步。Xcode 创建的任何组(group)都必须在文件系统有相应的映射。为了更清晰,代码不仅应该按照类型进行分组,也可以根据业务功能进行分组
  2. 合理组织工程的内的文件夹,工程中一般包括但不限于以下几个文件夹category(分类)、util/helper(工具类)、resource(资源)、const(常量)、third(第三方)
  3. 尽可能一直打开 target Build Settings 中 “Treat Warnings as Errors” 以及一些额外的警告。如果你需要忽略指定的警告,使用 Clang 的编译特性