专栏首页ios 技术积累iOS NSNotificationCenter通知中心的实现原理

iOS NSNotificationCenter通知中心的实现原理

NSNotificatinonCenter实现原理

  • NSNotificatinonCenter是使用观察者模式来实现的用于跨层传递消息,用来降低耦合度。
  • NSNotificatinonCenter用来管理通知,将观察者注册到NSNotificatinonCenter的通知调度表中,然后发送通知时利用标识符nameobject识别出调度表中的观察者,然后调用相应的观察者的方法,即传递消息(在Objective-C中对象调用方法,就是传递消息,消息有name或者selector,可以接受参数,而且可能有返回值),如果是基于block创建的通知就调用NSNotificationblock

NSNotification

NSNotification是方便NSNotificationCenter广播到其他对象时的封装对象,简单讲即通知中心对通知调度表中的对象广播时发送NSNotification对象。

NSNotification的工作机制

1.应用程序中需要订阅通知的对象,会向通告中心(Notification Center,NSNotificationCenter类的实例)注册,从而成为该事件的监听者。在注册过程中,监听者需要指定方法供通告中心在事件发生时调用。

2.监听对象发生变化后,对象给通告中心发一个通告(NSnotification的实例)。该通告对象包括识别通告的标志、发布通告的对象ID和可选的附加信息字典。

3.通告中心发送消息到每个已注册的监听者,调用监听者指定的方法会将通告传给这些监听者。

@interface NSNotification : NSObject <NSCopying, NSCoding>

@property (readonly, copy) NSNotificationName name;
@property (nullable, readonly, retain) id object;
@property (nullable, readonly, copy) NSDictionary *userInfo;

NSNotification对象包含:名称、object、字典 三个属性,名称是用来标识通知的标记,object是要通知的对象可以为nil,字典用来存储发送通知时附带的信息,也可以为nil

# NSNotificationCenter

NSNotificationCenter是类似一个广播中心站,使用defaultCenter来获取应用中的通知中心,它可以向应用任何地方发送和接收通知。在通知中心注册观察者,发送者使用通知中心广播时,以NSNotificationnameobject来确定需要发送给哪个观察者。为保证观察者能接收到通知,所以应先向通知中心注册观察者,接着再发送通知这样才能在通知中心调度表中查找到相应观察者进行通知。

发送者

发送通知可使用以下方法发送通知

- (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对象给通知中心注册的观察者。
  • 发送通知通过nameobject来确定来标识观察者,nameobject两个参数的规则相同即当通知设置namekChangeNotifition时,那么只会发送给符合namekChangeNotifition的观察者,同理object指发送给某个特定对象通知,如果只设置了name,那么只有对应名称的通知会触发。如果同时设置nameobject参数时就必须同时符合这两个条件的观察者才能接收到通知。

观察者

- (void)addObserver:(id)observer selector:(SEL)aSelector name:(nullable NSNotificationName)aName object:(nullable id)anObject;

注册anObserver对象:接受名字为notificationName, 发送者为anObject的notification. 当anObject发送名字为notificationName的notification时, 将会调用anObserver的aSelector方法

移除观察者

在对象被释放前需要移除掉观察者,避免已经被释放的对象还接收到通知导致崩溃。 移除观察者有两种方式:

- (void)removeObserver:(id)observer;
- (void)removeObserver:(id)observer name:(nullable NSNotificationName)aName object:(nullable id)anObject;
  • 传入相应的需要移除的observer 或者使用第二种方式三个参数来移除指定某个观察者。
  • 如果使用基于-[NSNotificationCenter addObserverForName:object:queue:usingBlock:]方法在获取方法返回的观察者进行释放。基于这个方法我们还可以让观察者接到通知后只执行一次:
__block __weak id<NSObject> observer = [[NSNotificationCenter defaultCenter] addObserverForName:kChangeNotifition object:nil queue:[NSOperationQueue mainQueue] usingBlock:^(NSNotification * _Nonnull note) {
       NSLog(@"-[NSNotificationCenter addObserverForName:object:queue:usingBlock:]");
      [[NSNotificationCenter defaultCenter] removeObserver:observer];
   }];

NSNotificationQueue(通知队列)

NSNotificationQueue是notification Center的缓冲池。如果我们使用普通的

- (void)postNotification:(NSNotification *)notification

这种方法来发送通知,那么这个通知就会直接发送到notification Center,notification Center则会直接将其发送给注册了该通知的观察者。但是如果我们使用NSNotificationQueue就不一样了,通知不是直接发送给notification Center,而是先发送给NSNotificationQueue,然后由NSNotificationQueue决定在当前runloop结束或者空闲的时候转发给notification Center,再由notification转发给注册的观察者。通过NSNotificationQueue,可以合并重复的通知,以便只发送一个通知。

NSNotificationQueue遵循FIFO的顺序,当一个通知移动到NSNotificationQueue的最前面,它就被发送给notification Center,然后notification Center再将通知转发给注册了该通知的监听者。 每一个线程都有一个默认的NSNotificationQueue,这个NSNotificationQueue和通知中心联系在一起。当然我们也可以自己创建NSNotificationQueue,可以为一个线程创建多个NSNotificationQueue。 NSNotificationQueue的核心方法有下列几个:

//类方法返回当前线程的默认的NSNotificationQueue。
defaultQueue
- (void)enqueueNotification:(NSNotification *)notification postingStyle:(NSPostingStyle)postingStyle coalesceMask:(NSNotificationCoalescing)coalesceMask forModes:(nullable NSArray<NSRunLoopMode> *)modes;

上面这个方法是使用NSNotificationQueue来发送通知用的。这里面有四个参数。

notification是所要发送的通知。 postingStyle 这是一个枚举类型的参数。

typedef NS_ENUM(NSUInteger, NSPostingStyle) {
    NSPostWhenIdle = 1,
    NSPostASAP = 2,
    NSPostNow = 3
};

NSPostingStyle即指发送通知的方式,一共有三种方式。

NSPostWhenIdle

通过字面意思大概可以知道是在空闲时发送。 简单地说就是当本线程的runloop空闲时即发送通知到通知中心。

NSPostASAP

ASAP即as soon as possible,就是说尽可能快。 当当前通知或者timer的回调执行完毕时发送通知到通知中心。

NSPostNow

多个相同的通知合并之后马上发送。

coalesceMask

coalesceMask即多个通知的合并方式。它也是一个枚举类型。 有时候会在一段时间内向NSNotificationQueue发送多个通知,有些通知是重复的,我们并不希望这些通知全部发送带通知中心,那么就可以使用这个枚举类型的参数。

typedef NS_OPTIONS(NSUInteger, NSNotificationCoalescing) {
    NSNotificationNoCoalescing = 0,
    NSNotificationCoalescingOnName = 1,
    NSNotificationCoalescingOnSender = 2
};
NSNotificationNoCoalescing

不管是否重复,不合并。

NSNotificationCoalescingOnName

按照通知的名字,如果名字重复,则移除重复的。

NSNotificationCoalescingOnSender

按照发送方,如果多个通知的发送方是一样的,则只保留一个。

modes

这里的mode指定的是当前的runloop的mode,指定mode后,只有当前线程的runloop在这个特定的mode下才能将通知发送到通知中心。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • Java 重载和重写的区别

    重载(overloading) 是在一个类里面,方法名字相同,而参数不同。返回类型可以相同也可以不同。

    赵哥窟
  • Swift 数组对象排序

    赵哥窟
  • ios 百度地图 获取拖动或缩放手势

    在项目中遇到一个问题,在拖动或者缩放百度地图的时候要请求数据。但是百度地图SDK中没有明确如何获取拖动和缩放手势 官方推荐使用如下两个方法,通过判断状态来获取...

    赵哥窟
  • WebView性能、体验分析与优化

    在App开发中,内嵌WebView始终占有着一席之地。它能以较低的成本实现Android、iOS和Web的复用,也可以冠冕堂皇的突破苹果对热更新的封锁。 然而便...

    美团技术团队
  • python基础-1:python模块概

                                 -------谢谢您的参考,如有疑问,欢迎交流

    py3study
  • [一对一讲什么] 之 测完了接口、搞好了目录,然后做啥?

    接 [一对一讲什么] 之 切图之后做啥?要加入项目和测试接口 上回书说,切完了页面,把HTML文件啊,CSS,JS,IMG图片什么都扔到相应的开发环境目录里,把...

    web前端教室
  • linux下安装python3

      wget https://www.python.org/ftp/python/3.6.0/Python-3.6.0a1.tar.xz

    py3study
  • WordPress自定义查询WP_Query使用方法大全

      自定义调用文章在网站建设中很常用,wordpress也很人性化,用新建查询new WP_Query就能实现相关功能。WP_Query怎么用呢?随ytkah一...

    ytkah
  • Hello World背后的故事:如何在Linux上编译C语言程序

    C语言的经典程序“Hello World”并不难写,很多朋友都可以闭着眼将它写出来。那么编译一个“Hello World”到底经历了怎样的过程呢?

    PP鲁
  • 工业物联网存在的意义到底是什么?

    随着工业物联网项目逐渐远离以云计算为中心的方法,人工智能和工业物联网演进的下一步将满足将算法转换为边缘计算工作的需求,而占用空间要小得多。

    万物皆可联i

扫码关注云+社区

领取腾讯云代金券