高效开发之NSNotificationCenter使用指南
看一遍就会哦!
最近又干回了我的老本行 - iOS,感觉自己的逻辑思维比以前严密了许多,但是也发现有很多知识没有掌握牢靠。在使用NSNotification时竟发现有些手生,赶紧抽时间整理一波,授人玫瑰,手有余香。
本文内容参考自官方API Reference
通知
Notification就是一个消息通知机制,类似广播。观察者向消息中心注册感兴趣的东西,当感兴趣的东西发出这个消息的时候,观察者就能接受到通知,然后去做相对应的事情。使用消息通知可以起到多个对象之间解耦的作用。苹果封装了常用的一些通知,比如窗口获得焦点、网络连接关闭等事件信息等。当然,我们也可以自定义个性化的通知。
通知中心,就是分发这些通知的地方,Cocoa框架中有两种通知中心:
- NSNotificationCenter 单进程通知管理
- NSDistributedNotificationCenter 一台机器中多个进程的通知管理
NSNotificationCenter
每个进程都包含一个默认的通知中心,这个通知中心是一个单例,通过[NSNotificationCenter defaultCenter]来获取。
通知中心将通知分发给观察者处理采用了同步机制,即当某一个通知被发送时,会一直阻塞在发送方法内,直到通知中心将该通知分发给注册过的观察者并且观察者完成了相应的处理之后,发送者才能继续执行其所在线程内的后续代码。如果要使用异步机制发送通知,文章后面会讲到“通知队列”,即第二种通知中心。
开发以来,感觉通知常用的场景有两个:
- 当前页面给上一个页面传值。
- 当一个值发生改变时,相应的对象作出相应的操作。
创建通知
- (instancetype)initWithName:(NSNotificationName)name object:(nullable id)object userInfo:(nullable NSDictionary *)userInfo;
+ (instancetype)notificationWithName:(NSNotificationName)aName object:(nullable id)anObject;
+ (instancetype)notificationWithName:(NSNotificationName)aName object:(nullable id)anObject userInfo:(nullable NSDictionary *)aUserInfo;
这三个方法都能成功的创建一个通知,然而创建通知并不是必要的操作,因为在发送通知时,可以包含通知的各项参数。但是单独讲一下可以加深理解嘛。ʘʚʘ
从上面三个方法可以看出,一个通知包含以下几个参数:
- name:通知的名字,是通知能够被接收到的关键因素。
- object:通知发送者希望发送给观察者的一个对象,通常为通知发送者自己。
- userInfo:一个字典,是通知发送者传递给观察者的参数。
发布通知
- (void)postNotification:(NSNotification *)notification;
- (void)postNotificationName:(NSNotificationName)aName object:(nullable id)anObject;
- (void)postNotificationName:(NSNotificationName)aName object:(nullable id)anObject userInfo:(nullable NSDictionary *)aUserInfo;
第一个方法适用于发布已经通过NSNotification创建好的通知,其余两个方法就在发布通知的时候,写入了通知的具体参数。
注册通知
- (void)addObserver:(id)observer selector:(SEL)aSelector name:(NSString *)aName object:(id)anObject;
在需要接收通知的位置注册通知,注册通知时需要设置:
- observer:观察者,通常是self。
- selector:收到通知后,回调监听器的这个方法,并且把通知对象当做参数传入。
- name:通知的名称,决定了观察者所接收的通知。如果为nil,那么观察者会接收到所有通知。
- object:通知的发布者。
注册完通知之后就可以静静等待猎物的到来了。
移除通知
- (void)removeObserver:(id)observer;
- (void)removeObserver:(id)observer name:(NSString *)aName object:(id)anObject;
移除通知是很重要的操作,因为通知中心是不会保留(retain)观察者对象的,在通知中心注册过的观察者,必须在观察者对象释放前取消注册。否则,当相应的通知再次出现时,通知中心仍然会向该观察者发送消息。因为相应的观察者对象已经被释放了,所以可能会导致应用崩溃。
取消注册的代码一般写在dealloc函数或者viewWillDisappear函数中。
举个栗子
光说不练假把式。下面展示一个简单的例子,在当前页面dimiss的时候通过notification给上一个页面传值。
ViewController1
- (void)viewWillAppear:(BOOL)animated {
[super viewWillAppear:animated];
//注册通知
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(notifyReceived:) name:@"notify" object:nil];
}
- (void)notifyReceived:(NSNotification *)content {
NSLog(@"-----接收到通知------");
NSLog(@"%@", content.userInfo[@"param"]);
}
- (void)viewWillDisappear:(BOOL)animated {
[super viewWillDisappear:animated];
//移除通知
[[NSNotificationCenter defaultCenter] removeObserve:self];
}
ViewController2
- (void)gotoNextVC {
//设置需要传递的参数
NSDictionary *dict =[[NSDictionary alloc] initWithObjectsAndKeys:@"我是一个参数",@"param", nil];
//创建通知
NSNotification *notification =[NSNotification notificationWithName:@"notify" object:nil userInfo:dict];
//通过通知中心发送通知
[[NSNotificationCenter defaultCenter] postNotification:notification];
[self dismissViewControllerAnimated:YES completion:nil];
}
NSDistributedNotificationCenter
分布式通知中心。这里就使用到了通知队列(Notification queue),通知队列通常以先进先出的顺序管理通知,当一个通知到达队首时,通知队列就将这个通知发送给通知中心,通知中心将其派发给所有监听它的观察者们。
每一个线程都有一个默认的通知队列,也是一个单例,通过[NSNotificationQueue defaultQueue]获取。我们也可以自定义通知队列,每个通知中心和线程可以有多个队列。
这个就不细讲了,如果下次有用到,会结合实际情况再进行总结的。
┌(ㆆ㉨ㆆ)ʃ有用的话,动动