iOS中Notification是同步还是异步的问题剖析

一、通知在主线程中默认是同步的

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         (立刻发送)
我们可以根据通知的紧急程度进行选择;

你可能感兴趣的:(iOS中Notification是同步还是异步的问题剖析)