如果在一個(gè)類中想要執(zhí)行另一個(gè)類中的方法可以使用通知
1.創(chuàng)建一個(gè)通知對(duì)象:使用notificationWithName:object: 或者 notificationWithName:object:userInfo:
NSNotification* notification = [NSNotification notificationWithName:kImageNotificationLoadFailed(connection.imageURL)
object:self
userInfo:[NSDictionary dictionaryWithObjectsAndKeys:error,@"error",connection.imageURL,@"imageURL",nil]];
這里需要注意的是砌庄,創(chuàng)建自己的通知并不是必須的。而是在創(chuàng)建自己的通知之前访递,采用NSNotificationCenter類的方法 postNotificationName:object: 和 postNotificationName:object:userInfo:更加便利的發(fā)出通知购裙。這種情況谱俭,一般使用NSNotificationCenter的類方法defaultCenter就獲得默認(rèn)的通知對(duì)象,這樣你就可以給該程序的默認(rèn)通知中心發(fā)送通知了。注意:每一個(gè)程序都有一個(gè)自己的通知中心奋刽,即NSNotificationCenter對(duì)象备典。該對(duì)象采用單例設(shè)計(jì)模式异旧,采用defaultCenter方法就可以獲得唯一的NSNotificationCenter對(duì)象。
注意:NSNotification對(duì)象是不可變的提佣,因?yàn)橐坏﹦?chuàng)建吮蛹,對(duì)象是不能更改的。
2.注冊(cè)通知:addObserver:selector:name:object:
可以看到除了添加觀察者之外拌屏,還有其接收到通知之后的執(zhí)行方法入口潮针,即selector的實(shí)參。因此為了進(jìn)行防御式編程倚喂,最好先檢查觀察者是否定義了該方法每篷。例如:添加觀察者代碼有
[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(aWindowBecameMain:)
name:NSWindowDidBecomeMainNotification object:nil];
這里保證了self定義了aWindowBecameMain:方法。而對(duì)于一個(gè)任意的觀察者observer端圈,不能保證其對(duì)應(yīng)的selector有aWindowBecameMain:焦读,可采用[observer respondsToSelector:@selector(aWindowBecameMain:)]] 進(jìn)行檢查。所以完整的添加觀察者過程為:
if([observer respondsToSelector:@selector(aWindowBecameMain:)]) {
[[NSNotificationCenter defaultCenter] addObserver:observer selector:@selector(aWindowBecameMain:) name:NSWindowDidBecomeMainNotification object:nil];
}
注意到addObserver:selector:name:object:不僅指定一個(gè)觀察者舱权,指定通知中心發(fā)送給觀察者的消息矗晃,還有接收通知的名字,以及指定的對(duì)象刑巧。一般來說不需要指定name和object喧兄,但如果僅僅指定了一個(gè)object无畔,觀察者將收到該對(duì)象的所有通知。例如將上面的代碼中name改為nil,那么觀察者將接收到object對(duì)象的所有消息吠冤,但是確定不了接收這些消息的順序浑彰。如果指指定一個(gè)通知名稱,觀察者將收到它每次發(fā)出的通知拯辙。例如郭变,上面的代碼中object為nil,那么客戶對(duì)象(self)將收到任何對(duì)象發(fā)出NSWindowDidBecomeMainNotification通知涯保。如果既沒有指定指定object诉濒,也沒有指定name,那么該觀察者將收到所有對(duì)象的所有消息夕春。
3.發(fā)送通知:postNotificationName:object:或者performSelectorOnMainThread:withObject:waitUntilDone:
例如程序可以實(shí)現(xiàn)將一個(gè)文本可以進(jìn)行一系列的轉(zhuǎn)換未荒,例如對(duì)于一個(gè)實(shí)例、RTF格式轉(zhuǎn)換成ASCII格式及志。而轉(zhuǎn)換在一個(gè)類(如Converter類)的對(duì)象中得到處理片排,在誠尋執(zhí)行過程中可以加入或者刪除這種轉(zhuǎn)換。而且當(dāng)添加或者刪除Converter操作時(shí)速侈,你的程序可能需要通知其他的對(duì)象率寡,但是這些Converter對(duì)象并不需要知道被通知對(duì)象是什么,能干什么倚搬。你只需要聲明兩個(gè)通知冶共,"ConverterAdded" 和 "ConverterRemoved",并且在某一事件發(fā)生時(shí)就發(fā)出這兩個(gè)通知每界。
當(dāng)一個(gè)用戶安裝或者刪除一個(gè)Converter捅僵,它將發(fā)送下面的消息給通知中心:
[[NSNotificationCenter defaultCenter]
postNotificationName:@"ConverterAdded" object:self];
或者是
[[NSNotificationCenter defaultCenter]
postNotificationName:@"ConverterRemoved" object:self];
通知中心將會(huì)區(qū)分它們對(duì)象對(duì)這些通知感興趣并且通知他們。如果除了關(guān)心觀察者的通知名稱和觀察的對(duì)象盆犁,還關(guān)心其他之外的對(duì)象命咐,那么就把之外的對(duì)象放在通知的可選字典中篡九,或者用方法postNotificationName:object:userInfo:谐岁。
而采用performSelectorOnMainThread:withObject:waitUntilDone:則是直接調(diào)用NSNotification的方法postNotification,而postNotificationName和object參數(shù)可以放到withObject的實(shí)參中榛臼。例如:
[[NSNotificationCenter defaultCenter] performSelectorOnMainThread:@selector(postNotification:) withObject:notification waitUntilDone:YES];//注意這里的notification為自定義的一個(gè)通知對(duì)象伊佃,可定義為NSNotification* notification = [NSNotification notificationWithName:@"ConverterAdded"object:self];//那么它的作用與上面的一致
4.移除通知:removeObserver:和removeObserver:name:object:
其中,removeObserver:是刪除通知中心保存的調(diào)度表一個(gè)觀察者的所有入口沛善,而removeObserver:name:object:是刪除匹配了通知中心保存的調(diào)度表中觀察者的一個(gè)入口航揉。
這個(gè)比較簡單,直接調(diào)用該方法就行金刁。例如:
[[NSNotificationCenter defaultCenter] removeObserver:observer name:nil object:self];
注意參數(shù)notificationObserver為要?jiǎng)h除的觀察者帅涂,一定不能置為nil议薪。
PS:這里簡單說一下通知中心保存的調(diào)度表。通知中心的調(diào)度表是給一些觀察者指定的一些通知集媳友。一個(gè)通知集是通知中心發(fā)出的通知的子集斯议。每個(gè)表的入口包含:
通知觀察者(必須要的)、通知名稱醇锚、通知的發(fā)送者哼御。
最后,提醒一下觀察者收到通知的順序是沒有定義的焊唬。同時(shí)通知發(fā)出和觀察的對(duì)象有可能是一樣的恋昼。通知中心同步轉(zhuǎn)發(fā)通知給觀察者,就是說 postNotification: 方法直到接收并處理完通知才返回值赶促。要想異步的發(fā)送通知液肌,可以使用NSNotificationQueue。在多線程編程中鸥滨,通知一般是在一個(gè)發(fā)出通知的那個(gè)線程中轉(zhuǎn)發(fā)矩屁,但也可能是不在同一個(gè)線程中轉(zhuǎn)發(fā)通知。