iOS中Notification是同步还是异步的问题剖析
2019-04-20 本文已影响0人
如日之升101
一、通知在主线程中默认是同步的
NSNotification默认在主线程中通知是同步的,当通知产生时,通知中心会一直等待所有的观察者都收到并且处理通知结束,才会返回到发送通知的地方继续执行后边的代码(所以处理通知的方法中如果存在耗时操作将会引起程序卡顿);
代码示例如下:
- (void)textNotifation {
#define XYNotificationTestName @"XYNotificationTestName"
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(progress) name:XYNotificationTestName object:nil];
NSLog(@"即将发出通知");
[[NSNotificationCenter defaultCenter] postNotificationName:XYNotificationTestName object:nil];
NSLog(@"处理发出通知的下一条代码");
}
- (void)progress {
sleep(10);
NSLog(@"通知处理结束");
}
打印出来结果为:
2019-04-20 13:58:52.004154+0800 TechnolgyTest[45755:5701088] 即将发出通知
2019-04-20 13:59:02.004102+0800 TechnolgyTest[45755:5701088] 通知处理结束
2019-04-20 13:59:02.004269+0800 TechnolgyTest[45755:5701088] 处理发出通知的下一条代码
以上示例证明通知在主线程中默认是同步的,这就要求我们在开发中要避免在通知的处理方法中添加耗时操作。
二、让通知异步的方法
1、将通知的发送放到子线程中
代码示例如下
NSLog(@"即将发出通知");
dispatch_async(dispatch_get_global_queue(0, 0), ^{
[[NSNotificationCenter defaultCenter] postNotificationName:XYNotificationTestName object:nil];
});
NSLog(@"处理发出通知的下一条代码");
这个时候控制台得到的结果为:
2019-04-20 14:23:56.403482+0800 TechnolgyTest[46411:5715874] 即将发出通知
2019-04-20 14:23:56.403639+0800 TechnolgyTest[46411:5715874] 处理发出通知的下一条代码
2019-04-20 14:24:06.404744+0800 TechnolgyTest[46411:5715945] 通知处理结束
由此我们可以得出结论: 将通知的发送方法防到子线程中可以使通知变成异步的。
2、将通知的处理方法放到子线程中
我们在主线程中post通知,同时将处理方法的代码放到子线程中,代码示例:
- (void)progress {
dispatch_async(dispatch_get_global_queue(0, 0), ^{
sleep(2);
NSLog(@"通知处理结束");
});
}
这个时候控制台得到的结果为:
2019-04-20 14:27:36.520127+0800 TechnolgyTest[46539:5718264] 即将发出通知
2019-04-20 14:27:36.520284+0800 TechnolgyTest[46539:5718264] 处理发出通知的下一条代码
2019-04-20 14:27:46.521147+0800 TechnolgyTest[46539:5718320] 通知处理结束
3、通知的发送可以添加到NSNotificationQueue异步通知缓冲队列中
我们将通知放到通知队列中发送,代码示例如下:
NSLog(@"即将发出通知");
NSNotification *notification = [NSNotification notificationWithName:XYNotificationTestName object:nil];
[[NSNotificationQueue defaultQueue] enqueueNotification:notification postingStyle:NSPostASAP];
NSLog(@"处理发出通知的下一条代码");
此时得到的结果为:
2019-04-20 14:30:40.302657+0800 TechnolgyTest[46639:5720748] 即将发出通知
2019-04-20 14:30:40.302846+0800 TechnolgyTest[46639:5720748] 处理发出通知的下一条代码
2019-04-20 14:30:42.313722+0800 TechnolgyTest[46639:5720748] 通知处理结束
由此我们得出结论: 可以将通知放入NSNotificationQueue中并指定为NSPostASAp或者NSPostWhenIdle类型发送,可以使的通知的处理变为异步;(注意指定NSPostNow模式立即发送通知,得到的结果仍然是同步执行)
注:NSNoticicationQueue
NSNoticicationQueue是一个通知缓冲队列,通常以FIFO(先进先出)的规则维护通知队列的发送。
向通知队列中添加通知有三种枚举类型:
NSPostASAp (尽快发送)
NSPostWhenIdle (空闲时发送)
NSPostNow (立刻发送)
我们可以根据通知的紧急程度进行选择;