iOS开发-消息传递方式-NotificationCenter

2018-03-08  本文已影响92人  张囧瑞

说完了target-action

说完了KVO

今天说另一种消息传递的方式,这种方式和KVO很像,通常也是用于一对多的情况,这种消息传递的方式就是NotificationCenter

NotificationCenter 翻译过来就是通知中心,他和我们生活中的广播很相似。

如何使用NotificationCenter

使用NotificationCenter和KVO非常相似,一般也是分为4个步骤:

一般在使用NotificationCenter时我们通常都是直接使用系统默认的NotificationCenter


@property (class, readonly, strong) NSNotificationCenter *defaultCenter;

使用类方法调用他之后,我们就可以做响应的添加、发出通知、移除等操作了。

添加观察者

添加观察者有两种方式,一个是直接调用函数添加,然后再编写实现的方法,通过selector来响应通知。

还有一种方法是使用block,在代码块内部实现响应通知的操作。

直接添加

直接添加观察者时可以通过函数:


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

传递了响应的参数之后就完成了观察者的添加,那我们来看看传递的参数都有什么作用。

observer

通知的观察者,当通知post出来之后,观察者会收到响应的通知,不可以为nil。

aSelector

收到通知之后需要做的操作,当通知post出来之后,会触发该方法,当中会传递一个NSNotification的参数。

aName

通知名称,用来区分发送通知的对象,可以为nil,但是因为应用的执行的过程中会发出很多通知,如果为nil则代表接受所有的通知。

anObject

发送的对象,也是用来区分发送通知的对象,可以为nil,当为nil时,也会接受同一通知名称的所有的通知。

一般情况下,如果我们将通知名称和发送对象都置为nil时,我们可以监听到当前通知中心的全部通知。比如:


- (void)addNotificationCenter

{

 [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(notificationCall:) name:nil object:nil];

 [[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:nil];

}

- (void)notificationCall:(NSNotification *)notification

{

 NSLog(@"notification = %@", notification.name);

}

上边一段代码执行了之后我们会发现结果如下:


2018-03-08 17:28:25.368264+0800 MessagePassingDemo[50635:3561130] notification = postNotification

2018-03-08 17:28:25.368593+0800 MessagePassingDemo[50635:3561130] notification = UINavigationControllerWillShowViewControllerNotification

2018-03-08 17:28:25.370494+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification

2018-03-08 17:28:25.370677+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification

2018-03-08 17:28:25.371431+0800 MessagePassingDemo[50635:3561130] notification = _UIWindowContentWillRotateNotification

2018-03-08 17:28:25.372607+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification

2018-03-08 17:28:25.372898+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification

2018-03-08 17:28:25.373097+0800 MessagePassingDemo[50635:3561130] notification = _UIWindowContentWillRotateNotification

2018-03-08 17:28:25.392586+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification

2018-03-08 17:28:25.393403+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification

2018-03-08 17:28:25.423827+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification

2018-03-08 17:28:25.424067+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification

2018-03-08 17:28:25.424322+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification

2018-03-08 17:28:25.424721+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification

2018-03-08 17:28:25.426606+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification

2018-03-08 17:28:25.426780+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification

2018-03-08 17:28:25.427223+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification

2018-03-08 17:28:25.427401+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification

2018-03-08 17:28:25.431615+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification

2018-03-08 17:28:25.790567+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification

2018-03-08 17:28:25.790769+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification

2018-03-08 17:28:25.791427+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification

2018-03-08 17:28:25.792659+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidCommitNotification

2018-03-08 17:28:25.793148+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification

2018-03-08 17:28:25.935701+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification

2018-03-08 17:28:25.936635+0800 MessagePassingDemo[50635:3561130] notification = UINavigationControllerDidShowViewControllerNotification

2018-03-08 17:28:25.936938+0800 MessagePassingDemo[50635:3561130] notification = UIViewAnimationDidStopNotification

所以一般情况下,添加观察者时,我们至少都会指定好通知的名称。


- (void)addNotificationCenter

{

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

 [[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:nil];

}

- (void)notificationCall:(NSNotification *)notification

{

 NSLog(@"notification = %@", notification.name);

}

输出:


2018-03-08 17:30:22.767448+0800 MessagePassingDemo[50709:3564576] notification = postNotification

另外,如果我们设置了通知名,没有设置发送对象时,所有此通知名的通知都会被接收到,如果设置了发送对象:


- (void)addNotificationCenter

{

 [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(notificationCall:) name:@"postNotification" object:self.buttonA];    

}

- (void)buttonAClicked

{

 [[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:self.buttonA];

}

- (void)buttonBClicked

{

 [[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:self.buttonB];

}

这时候就只有buttonAClickedbuttonA被点击时才会触发通知。

使用 block

使用block添加观察者时可以调用如下函数:


- (id <NSObject>)addObserverForName:(nullable NSNotificationName)name object:(nullable id)obj queue:(nullable NSOperationQueue *)queue usingBlock:(void (^)(NSNotification *note))block API_AVAILABLE(macos(10.6), ios(4.0), watchos(2.0), tvos(9.0));

相比较于上一个方法,这种方法中没有指定具体的观察者,并且用block代替了触发的方法,而且新增了一个queue

因为用block回调来替代了触发的方法,所以在发送通知时,也不需要去某个具体观察者的方法列表中去找对应的列表了,直接执行block中的内容就可以了。

比如上边的方法,我们就可以改成这样:


 [[NSNotificationCenter defaultCenter] addObserverForName:@"postNotification" object:nil queue:nil usingBlock:^(NSNotification * _Nonnull note) {

 NSLog(@"notification = %@", note.name);

 }];

 [[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:nil];

这样写,可以让代码看起来更加紧凑。

而新增的queue,在默认情况下在post的线程中处理,比如这段代码:


- (void)addNotificationCenter

{    

 [[NSNotificationCenter defaultCenter] addObserverForName:@"postNotification" object:nil queue:nil usingBlock:^(NSNotification * _Nonnull note) {

 NSLog(@"notification = %@", note.name);

 NSLog(@"receive thread = %@", [NSThread currentThread]);

 }];

 [[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:nil];

 dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{

 NSLog(@"post thread = %@", [NSThread currentThread]);

 [[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:nil];

 });

}

打印出来我们就会发现:


2018-03-08 17:49:54.732746+0800 MessagePassingDemo[51217:3595209] notification = postNotification

2018-03-08 17:49:54.733093+0800 MessagePassingDemo[51217:3595209] receive thread = <NSThread: 0x60000006e780>{number = 1, name = main}

2018-03-08 17:49:54.733711+0800 MessagePassingDemo[51217:3595267] post thread = <NSThread: 0x6000004639c0>{number = 3, name = (null)}

2018-03-08 17:49:54.734052+0800 MessagePassingDemo[51217:3595267] notification = postNotification

2018-03-08 17:49:54.734320+0800 MessagePassingDemo[51217:3595267] receive thread = <NSThread: 0x6000004639c0>{number = 3, name = (null)}

除了这部分的不同之外,剩下的两者应该没有什么不同的地方了,还有注意一点就是第二种方法中的循环引用问题的发生。

实现响应方法

响应方法这边就不多说了,一种是通过调用,一种是通过block,但是两种都会传递一个NSNotification对象,注意不是NSNotificationCenter,是NSNotification

其中包括了


@property (readonly, copy) NSNotificationName name;

@property (nullable, readonly, retain) id object;

@property (nullable, readonly, copy) NSDictionary *userInfo;

一般情况下,我们可以通过userInfo来传递一些信息。

post 通知

post通知一共有三种方法。


- (void)postNotification:(NSNotification *)notification;

- (void)postNotificationName:(NSNotificationName)aName object:(nullable id)anObject;

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

同理,name和object也可以为nil,只不过发出去的消息并没有理你罢了。另外userInfo用来传递一些简单的信息。

比如:


- (void)addNotificationCenter

{

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

 [[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:nil userInfo:@{@"name":@"zhangsan",@"age":@"18"}];

}


- (void)notificationCall:(NSNotification *)notification

{

 NSLog(@"notification = %@", notification.name);

 NSLog(@"userinfo = %@",notification.userInfo);

}

这边就会输出:


2018-03-08 17:58:14.998128+0800 MessagePassingDemo[51452:3608470] notification = postNotification

2018-03-08 17:58:14.998407+0800 MessagePassingDemo[51452:3608470] userinfo = {

 age = 18;

 name = zhangsan;

}

移除观察者

移除观察者一共有两种方法:


- (void)removeObserver:(id)observer;

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

第一个方法是不针对某个特定通知,移除全部观察者,第二种是有针对性的移除某个观察者。而如果在第二种方法中name和object参数都传递nil,就和第一种方法完全一样了。

一般在dealloc中会使用第一种方法移除全部的观察者。而在viewWillDisappear中则使用第二种方式移除。

这里注意,如果我们不在合适的时机移除观察者,导致添加了重复的观察者的话,新注册的已有名字的观察者并不会覆盖之前的观察者,而是会添加两个观察者,这会导致post时响应两次或更多,比如:


- (void)addNotificationCenter

{

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

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

 [[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:nil userInfo:@{@"name":@"zhangsan",@"age":@"18"}];

}


- (void)removeNotificationCenter

{

 [[NSNotificationCenter defaultCenter] removeObserver:self];

 [[NSNotificationCenter defaultCenter] removeObserver:self name:nil object:nil];

}

打印的结果:


2018-03-08 21:27:30.566903+0800 MessagePassingDemo[52218:3651551] notification = postNotification

2018-03-08 21:27:30.567124+0800 MessagePassingDemo[52218:3651551] userinfo = {

 age = 18;

 name = zhangsan;

}

2018-03-08 21:27:30.567253+0800 MessagePassingDemo[52218:3651551] notification = postNotification

2018-03-08 21:27:30.567440+0800 MessagePassingDemo[52218:3651551] userinfo = {

 age = 18;

 name = zhangsan;

}

最后

NotificationCenter的简单使用就这么多,以上这些内容仅供个人学习使用,如果有什么不对的地方还请各位大佬多多指教。

上一篇下一篇

猜你喜欢

热点阅读