如果在一個(gè)類中想要執(zhí)行另一個(gè)類中的方法可以使用通知
1.創(chuàng)建一個(gè)通知對象:使用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)的通知對象,這樣你就可以給該程序的默認(rèn)通知中心發(fā)送通知了。注意:每一個(gè)程序都有一個(gè)自己的通知中心,即NSNotificationCenter對象。該對象采用單例設(shè)計(jì)模式,采用defaultCenter方法就可以獲得唯一的NSNotificationCenter對象。
注意:NSNotification對象是不可變的,因?yàn)橐坏﹦?chuàng)建,對象是不能更改的。
2.注冊通知:addObserver:selector:name:object:
可以看到除了添加觀察者之外,還有其接收到通知之后的執(zhí)行方法入口,即selector的實(shí)參。因此為了進(jìn)行防御式編程,最好先檢查觀察者是否定義了該方法。例如:添加觀察者代碼有
[[NSNotificationCenter defaultCenter] addObserver:self
selector:@selector(aWindowBecameMain:)
name:NSWindowDidBecomeMainNotification object:nil];
這里保證了self定義了aWindowBecameMain:方法。而對于一個(gè)任意的觀察者observer,不能保證其對應(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ā)送給觀察者的消息,還有接收通知的名字,以及指定的對象。一般來說不需要指定name和object,但如果僅僅指定了一個(gè)object,觀察者將收到該對象的所有通知。例如將上面的代碼中name改為nil,那么觀察者將接收到object對象的所有消息,但是確定不了接收這些消息的順序。如果指指定一個(gè)通知名稱,觀察者將收到它每次發(fā)出的通知。例如,上面的代碼中object為nil,那么客戶對象(self)將收到任何對象發(fā)出NSWindowDidBecomeMainNotification通知。如果既沒有指定指定object,也沒有指定name,那么該觀察者將收到所有對象的所有消息。
3.發(fā)送通知:postNotificationName:object:或者performSelectorOnMainThread:withObject:waitUntilDone:
例如程序可以實(shí)現(xiàn)將一個(gè)文本可以進(jìn)行一系列的轉(zhuǎn)換,例如對于一個(gè)實(shí)例、RTF格式轉(zhuǎn)換成ASCII格式。而轉(zhuǎn)換在一個(gè)類(如Converter類)的對象中得到處理,在誠尋執(zhí)行過程中可以加入或者刪除這種轉(zhuǎn)換。而且當(dāng)添加或者刪除Converter操作時(shí),你的程序可能需要通知其他的對象,但是這些Converter對象并不需要知道被通知對象是什么,能干什么。你只需要聲明兩個(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];
通知中心將會區(qū)分它們對象對這些通知感興趣并且通知他們。如果除了關(guān)心觀察者的通知名稱和觀察的對象,還關(guān)心其他之外的對象,那么就把之外的對象放在通知的可選字典中,或者用方法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è)通知對象,可定義為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為要刪除的觀察者,一定不能置為nil。
PS:這里簡單說一下通知中心保存的調(diào)度表。通知中心的調(diào)度表是給一些觀察者指定的一些通知集。一個(gè)通知集是通知中心發(fā)出的通知的子集。每個(gè)表的入口包含:
通知觀察者(必須要的)、通知名稱、通知的發(fā)送者。
最后,提醒一下觀察者收到通知的順序是沒有定義的。同時(shí)通知發(fā)出和觀察的對象有可能是一樣的。通知中心同步轉(zhuǎn)發(fā)通知給觀察者,就是說 postNotification: 方法直到接收并處理完通知才返回值。要想異步的發(fā)送通知,可以使用NSNotificationQueue。在多線程編程中,通知一般是在一個(gè)發(fā)出通知的那個(gè)線程中轉(zhuǎn)發(fā),但也可能是不在同一個(gè)線程中轉(zhuǎn)發(fā)通知。