通知NSNotificationCenter详解(一)

版本记录

版本号 时间
V1.0 2017.07.17

前言

与代理相比,通知具有低耦合的特点,两个毫无关联的控制器之间可以进行相互通信并相应操作,用了通知很久,但是从没有系统的总结一下通知相关的知识和遇到的坑,下面就讲一下通知的相关知识,先从简单的知识点和API开始。

通知基础

通知是ios中的一种消息机制,观察者只要向消息中心注册, 即可接受其他对象发送来的消息,消息发送者和消息接受者两者可以互相一无所知,完全解耦。这种消息通知机制可以应用于任意时间和任何对象,观察者可以有多个,所以消息具有广播的性质,只是需要注意的是,观察者向消息中心注册以后,在不需要接受消息时需要向消息中心注销,这种消息广播机制是典型的“Observer”(观察者)模式。

消息机制常常用于在向服务器端请求数据或者提交数据的场景,在和服务器端成功交互后,需要处理服务器端返回的数据,或发送响应消息等,就需要用到消息机制。其原理主要如下所示:

通知原理

API

下面我们看一下ios中通知相关的API。

NSNotification

/****************   Notifications   ****************/

@interface NSNotification : NSObject <NSCopying, NSCoding>

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

- (instancetype)initWithName:(NSNotificationName)name object:(nullable id)object userInfo:(nullable NSDictionary *)userInfo NS_AVAILABLE(10_6, 4_0) NS_DESIGNATED_INITIALIZER;
- (nullable instancetype)initWithCoder:(NSCoder *)aDecoder NS_DESIGNATED_INITIALIZER;

@end

@interface NSNotification (NSNotificationCreation)

+ (instancetype)notificationWithName:(NSNotificationName)aName object:(nullable id)anObject;
+ (instancetype)notificationWithName:(NSNotificationName)aName object:(nullable id)anObject userInfo:(nullable NSDictionary *)aUserInfo;

- (instancetype)init /*NS_UNAVAILABLE*/;    /* do not invoke; not a valid initializer for this class */

@end

这里面有几点:

  • name:通知的名字,其实就是为通知起的别名,作用就是用以区分不同通知,如果为nil标识可以接收任何名字的通知。
  • object :发送通知的对象,用以区分是谁发出的通知,如果为nil,表示可以接受任何对象发送的通知。
  • userInfo:它是一个字典用来存储发送的通知信息,当我们post一个通知的时候,就可以存放在字典里面,add监听端,就可以将这个userInfo取出来,得到通知的内容。
  • 接下里的几个对象方法和类方法中存储的就是通知的初始化或者实例化方法。

** NSNotificationCenter**

/****************   Notification Center ****************/

@interface NSNotificationCenter : NSObject {
    @package
    void *_impl;
    void *_callback;
    void *_pad[11];
}

#if FOUNDATION_SWIFT_SDK_EPOCH_AT_LEAST(8)
@property (class, readonly, strong) NSNotificationCenter *defaultCenter;

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

- (void)postNotification:(NSNotification *)notification;
- (void)postNotificationName:(NSNotificationName)aName object:(nullable id)anObject;
- (void)postNotificationName:(NSNotificationName)aName object:(nullable id)anObject userInfo:(nullable NSDictionary *)aUserInfo;

- (void)removeObserver:(id)observer;
- (void)removeObserver:(id)observer name:(nullable NSNotificationName)aName object:(nullable id)anObject;

- (id <NSObject>)addObserverForName:(nullable NSNotificationName)name object:(nullable id)obj queue:(nullable NSOperationQueue *)queue usingBlock:(void (^)(NSNotification *note))block NS_AVAILABLE(10_6, 4_0);
    // The return value is retained by the system, and should be held onto by the caller in
    // order to remove the observer with removeObserver: later, to stop observation.

@end

这里涉及到的就是通知的转发、监听以及移除等,这个后续会详细说明。


通知的发送

有关通知发送的方法主要有三个,如下:

- (void)postNotification:(NSNotification *)notification;
- (void)postNotificationName:(NSNotificationName)aName object:(nullable id)anObject;
- (void)postNotificationName:(NSNotificationName)aName object:(nullable id)anObject userInfo:(nullable NSDictionary *)aUserInfo;

我们先说一下相关参数,这里的参数和注册观察者时候的参数也都是一样的,说一遍就不多说了。

  • notification:通知对象,通知中心发出的对象就是它。
  • aName:通知的名字,用以区分不同通知,在注册观察者的时候,传nil表示可以监听任何名称的通知。
  • anObject:发送通知的对象,如果传nil表示不关心是谁发送的通知,观察者那边传入nil表示可以接受任何发送者的通知。
  • aUserInfo:通知消息,是一个字典,发送的内容可以写在里面。

第一种发送通知

    //第一种发送通知
    NSNotification *notification = [NSNotification notificationWithName:@"notificationTestOne" object:self];
    [[NSNotificationCenter defaultCenter] postNotification:notification];
  

这里的object也可以为nil,表示不区分是谁发送的。

第二种发送通知

    //第二种发送通知
    [[NSNotificationCenter defaultCenter] postNotificationName:@"notificationTestTwo" object:self];

同上,就不多说了。

第三种发送通知

    //第三种发送通知
    NSDictionary *notiDict = @{@"key":@"content"};
    [[NSNotificationCenter defaultCenter] postNotificationName:@"notificationTestThree" object:self userInfo:notiDict];

这里,当没有什么传递的时候,userInfo可以为空,为空的时候就和前两种发送通知是一样的了。


通知的监听

通知是一种观察者模式,有人发送就要有人observer,并调用相关方法,处理相关逻辑。下面我们就接受一下上面三种发送的通知。

第一种监听通知

 [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(processTestOne:) name:@"notificationTestOne" object:nil];
- (void)processTestOne:(NSNotification *)noti
{
    NSLog(@"notificationTestOne");
}

我们看输出

2017-07-17 16:18:57.708 JJNotification[17071:1508348] notificationTestOne

可见,可以收到通知。

第二种监听通知

[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(processTestTwo:) name:@"notificationTestTwo" object:nil];

- (void)processTestTwo:(NSNotification *)noti
{
    NSLog(@"notificationTestTwo");
}

下面看输出

2017-07-17 16:24:20.773 JJNotification[17096:1537087] notificationTestTwo

可见,可以收到通知。

第三种监听通知

[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(processTestThree:) name:@"notificationTestThree" object:nil];
- (void)processTestThree:(NSNotification *)noti
{
    NSDictionary *notiDict = noti.userInfo;
    NSString *content = [notiDict objectForKey:@"key"];
    NSLog(@"notificationTestThree = %@",content);
}

下面看输出

2017-07-17 16:24:20.773 JJNotification[17096:1537087] notificationTestThree = content

可见,还是可以监听到通知。


通知的移除

不用的监听通知要移除,移除通知一般都是在delloc中完成,在delloc中可以移除全部监听的通知,还可以移除指定名字的通知。

移除全部通知


- (void)dealloc
{
    [[NSNotificationCenter defaultCenter] removeObserver:self];
}

如果这么写,那么上面三个notificationTestOnenotificationTestTwonotificationTestThree通知都会被移除,可谓一劳永逸。

移除指定名字的通知

但是有的时候的需求是只要移除指定的某个通知,而保留某一个或者几个通知,具体如下所示:

- (void)dealloc
{
    [[NSNotificationCenter defaultCenter] removeObserver:self name:@"notificationTestOne" object:nil];
}

这样子就把名字为notificationTestOne的通知移除了,其他的通知还会起作用,保留着监听的功能。

后记

这里写的就是ios API以及通知的监听、发送以及移除等基本知识,目的就是给新手能看明白,后面的几篇我会加入稍微难一点的东西还有就是一些通知过程中碰到的坑。

风景
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,332评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,508评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,812评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,607评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,728评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,919评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,071评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,802评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,256评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,576评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,712评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,389评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,032评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,798评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,026评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,473评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,606评论 2 350

推荐阅读更多精彩内容

  • 概述 在多数移动应用中任何时候都只能有一个应用程序处于活跃状态,如果其他应用此刻发生了一些用户感兴趣的那么通过通知...
    莫离_焱阅读 6,498评论 1 8
  • 点击查看原文 Web SDK 开发手册 SDK 概述 网易云信 SDK 为 Web 应用提供一个完善的 IM 系统...
    layjoy阅读 13,708评论 0 15
  • 转载自南峰子的技术博客 一个NSNotificationCenter对象(通知中心)提供了在程序中广播消息的机制,...
    我消失1314阅读 880评论 0 2
  • NSNotificationCenter对象(通知中心)提供了在程序中广播消息的机制,它实质上就是一个通知分发表。...
    9de75b652cd9阅读 751评论 0 1
  • 一个NSNotificationCenter对象(通知中心)提供了在程序中广播消息测机制,它实质上就是一个通知分发...
    DomAndMona阅读 807评论 0 2