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

说完了target-action
说完了KVO

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

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

如何使用NotificationCenter

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

  • 1.添加观察者。
  • 2.实现响应方法。
  • 3.发出通知。
  • 4.移除观察者。

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

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

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

添加观察者

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

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

直接添加

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

1
- (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时,我们可以监听到当前通知中心的全部通知。比如:

1
2
3
4
5
6
7
8
9
10
- (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);
}

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
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

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

1
2
3
4
5
6
7
8
9
10
11
- (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);
}

输出:

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

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
- (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添加观察者时可以调用如下函数:

1
- (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中的内容就可以了。

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

1
2
3
4
5
6
[[NSNotificationCenter defaultCenter] addObserverForName:@"postNotification" object:nil queue:nil usingBlock:^(NSNotification * _Nonnull note) {
NSLog(@"notification = %@", note.name);
}];
[[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:nil];

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

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
- (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];
});
}

打印出来我们就会发现:

1
2
3
4
5
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

其中包括了

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

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

post 通知

post通知一共有三种方法。

1
2
3
- (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用来传递一些简单的信息。

比如:

1
2
3
4
5
6
- (void)addNotificationCenter
{
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(notificationCall:) name:@"postNotification" object:nil];
[[NSNotificationCenter defaultCenter] postNotificationName:@"postNotification" object:nil userInfo:@{@"name":@"zhangsan",@"age":@"18"}];
}
1
2
3
4
5
6
- (void)notificationCall:(NSNotification *)notification
{
NSLog(@"notification = %@", notification.name);
NSLog(@"userinfo = %@",notification.userInfo);
}

这边就会输出:

1
2
3
4
5
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;
}

移除观察者

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

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

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

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

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

1
2
3
4
5
6
7
8
- (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"}];
}
1
2
3
4
5
6
- (void)removeNotificationCenter
{
[[NSNotificationCenter defaultCenter] removeObserver:self];
[[NSNotificationCenter defaultCenter] removeObserver:self name:nil object:nil];
}

打印的结果:

1
2
3
4
5
6
7
8
9
10
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的简单使用就这么多,以上这些内容仅供个人学习使用,如果有什么不对的地方还请各位大佬多多指教。