模拟新闻的发布:
// 1.初始化机构
NewsCompany *tx = [[NewsCompany alloc] init];
tx.name = @"腾讯新闻";
NewsCompany *sina = [[NewsCompany alloc] init];
sina.name = @"新浪新闻";
通知的发布者
====================================================
// 2.初始化3个人
Person *zhangsan = [[Person alloc] init];
zhangsan.name = @"张三";
Person *lisi = [[Person alloc] init];
lisi.name = @"李四";
Person *wangwu = [[Person alloc] init];
wangwu.name = @"王五";
通知的接受者(observer)
=================================
NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
通知中心===================================
通知对象(通知的发布者) observer (通知的接收者)
1.创建一个通知对象:使用notificationWithName:object: 或者 notificationWithName:object:userInfo:但是在实际的开发中,我们一般
不这样做。
那么怎么创建通知呢?
采用NSNotificationCenter类的方法 postNotificationName:object: 和 postNotificationName:object:userInfo:
创建通知,并将通知单发出。
比如:
// tx发布了一则叫做junshi_news_come的通知
[center postNotificationName:@"junshi_news_come"
object:tx
userInfo:@{@"title" : @"伊拉克战争停止了",
@"intro" : @"伊拉克战争停止了.........."}];
// sina发布了一则叫做junshi_news_come的通知
[center postNotificationName:@"yule_news_come"
object:sina
userInfo:@{@"title" : @"6456456456456",
@"intro" : @"7657567567567"}];
这样就创建并发布了两个通知。注意:每一个程序都有一个自己的通知中心,即NSNotificationCenter对象。该对象采用单例设计模式,采用defaultCenter方法就可以获得唯一的NSNotificationCenter对象。
注意:NSNotification对象是不可变的,因为一旦创建,对象是不能更改的。
2.注册通知:addObserver:selector:name:object:(添加监听器)
可以看到除了添加观察者之外,还有其接收到通知之后的执行方法入口,即selector的实参。
比如:
[center addObserver:zhangsan selector:@selector(newsCome:) name:@"junshi_news_come" object:nil];
[center addObserver:zhangsan selector:@selector(newsCome:) name:@"yule_news_come" object:nil];
//newsCome函数的实现,参数是一个通知NSNotification
- (void)newsCome:(NSNotification *)note
{
// 通知的发布者
NewsCompany *obj = note.object;
NSLog(@"%@接收到了%@发出的通知,通知内容是:%@", self.name, obj.name, note.userInfo);
}
注意到
addObserver://此参数是观察者对象(既通知的接收者)
selector://回调函数,观察者接收到通知要干什么事情
name://通知的名称
object://通知对象(谁发的通知)
一般来说不需要指定name和object,但如果仅仅指定了一个object,观察者将收到该对象的所有通知。例如将上面的代码中name改为nil,那么观察者将接收到object对象的所有消息,但是确定不了接收这些消息的顺序。如果指指定一个通知名称,观察者将收到它每次发出的通知。例如,上面的代码中object为nil,那么客户对象(self)将收到任何对象发出通知。如果既没有指定指定object,也没有指定name,那么该观察者将收到所有对象的所有消息。
3.发送通知:postNotificationName:object:或者performSelectorOnMainThread:withObject:waitUntilDone:
例如程序可以实现将一个文本可以进行一系列的转换,例如对于一个实例、RTF格式转换成ASCII格式。而转换在一个类(如Converter类)的对象中得到处理,在诚寻执行过程中可以加入或者删除这种转换。而且当添加或者删除Converter操作时,你的程序可能需要通知其他的对象,但是这些Converter对象并不需要知道被通知对象是什么,能干什么。你只需要声明两个通知,"ConverterAdded" 和 "ConverterRemoved",并且在某一事件发生时就发出这两个通知。
当一个用户安装或者删除一个Converter,它将发送下面的消息给通知中心:
[[NSNotificationCenter defaultCenter]
postNotificationName:@"ConverterAdded" object:self];
或者是
[[NSNotificationCenter defaultCenter]
postNotificationName:@"ConverterRemoved" object:self];
通知中心将会区分它们对象对这些通知感兴趣并且通知他们。如果除了关心观察者的通知名称和观察的对象,还关心其他之外的对象,那么就把之外的对象放在通知的可选字典中,或者用方法postNotificationName:object:userInfo:。
而采用performSelectorOnMainThread:withObject:waitUntilDone:则是直接调用NSNotification的方法postNotification,而postNotificationName和object参数可以放到withObject的实参中。例如:
[[NSNotificationCenter defaultCenter] performSelectorOnMainThread:@selector(postNotification:) withObject:notification waitUntilDone:YES];//注意这里的notification为自定义的一个通知对象,可定义为NSNotification* notification = [NSNotification notificationWithName:@"ConverterAdded"object:self];//那么它的作用与上面的一致
4.移除通知:removeObserver:和removeObserver:name:object:
其中,removeObserver:是删除通知中心保存的调度表一个观察者的所有入口,而removeObserver:name:object:是删除匹配了通知中心保存的调度表中观察者的一个入口。
这个比较简单,直接调用该方法就行。例如:
[[NSNotificationCenter defaultCenter] removeObserver:observer name:nil object:self];
注意参数notificationObserver为要删除的观察者,一定不能置为nil。
最后,提醒一下观察者收到通知的顺序是没有定义的。同时通知发出和观察的对象有可能是一样的。通知中心同步转发通知给观察者,就是说 postNotification: 方法直到接收并处理完通知才返回值。要想异步的发送通知,可以使用NSNotificationQueue。在多线程编程中,通知一般是在一个发出通知的那个线程中转发,但也可能是不在同一个线程中转发通知。