通常我們?cè)?iOS 中發(fā)生什么事件時(shí)該做什么是由 Delegate 實(shí)現(xiàn)的映凳,例如 View 加載完后會(huì)觸發(fā) viewDidLoad近上。 Apple 還為我們提供了另一種通知響應(yīng)方式贵试,那就是 NSNotification歌粥,系統(tǒng)中(UIKeyboardDidShowNotification 等) 以及某些第三方組件(例如 ASIHTTPRequest 的 kReachabilityChangedNotification 等)沮脖。
NSNotificationCenter較之于 Delegate 可以實(shí)現(xiàn)更大的跨度的通信機(jī)制金矛,可以為兩個(gè)無引用關(guān)系的兩個(gè)對(duì)象進(jìn)行通信。NSNotificationCenter 的通信原理使用了觀察者模式:
1. NSNotificationCenter 注冊(cè)觀察者對(duì)某個(gè)事件(以字符串命名)感興趣勺届,及該事件觸發(fā)時(shí)該執(zhí)行的 Selector 或 Block
2. NSNotificationCenter 在某個(gè)時(shí)機(jī)激發(fā)事件(以字符串命名)
3. 觀察者在收到感興趣的事件時(shí)驶俊,執(zhí)行相應(yīng)的 Selector 或 Block
使用 NSNotificationCenter 的步驟示例代碼:
1. 定義一個(gè)事件到來時(shí)該執(zhí)行的方法:
- (void)execute:(NSNotification *)notification {
//do something when received notification
//notification.name is @"NOTIFICATION_NAME"
if(notification.object && [notification.object isKindOfClass:[Test class]]){
//do something
}
}
2. 注冊(cè)觀察者:
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(execute:) name:@"NOTIFICATION_NAME" object:nil];
使用默認(rèn)的通知中心,上面代碼的意義的免姿,觀察者 self 在收到名為 @"NOTIFICATION_NAME" 的事件是執(zhí)行 @selector(execute:)饼酿,最后一個(gè)參數(shù)是表示會(huì)對(duì)哪個(gè)發(fā)送者對(duì)象發(fā)出的事件作出響應(yīng),nil 時(shí)表示接受所有發(fā)送者的事件。
還有一種注冊(cè)觀察者的方式是用方法:
- (id)addObserverForName:(NSString)nameobject:(id)objqueue:(NSOperationQueue)queueusingBlock:(void (^)(NSNotification ))block*
3. 激發(fā)事件故俐,即通知相應(yīng)的觀察者
[[NSNotificationCenter defaultCenter] postNotificationName:@"NOTIFICATION_NAME" object:nil];
//或者用下面幾行代碼想鹰,明確的 notification 示例
Test *test = [[Test alloc] init];
NSNotification *notification = [NSNotification notificationWithName:@"NOTIFICATION_NAME"
object:test];
[[NSNotificationCenter defaultCenter] postNotification:notification];
這里的 object 參數(shù)對(duì)應(yīng)到方法- (void)execute:(NSNotification *)notification里的notification.object, name 就是notification.name。
代碼到這里药版,方法- (void)execute:(NSNotification *)notification就會(huì)得到執(zhí)行了辑舷。
說明:我們上面用的[NSNotificationCenter defaultCenter]同一個(gè)實(shí)例,你也可以使用自己的NSNotificationCenter槽片,如:
NSNotificationCenter *notificationCenter = [[NSNotificationCenter alloc]init];
事件只會(huì)在當(dāng)前的 NotificationCenter 中廣播何缓,不同的 NotificationCenter 之間的事件通知互不相干。
NSNotificationCenter 比之 Delegate 的好處就是事件發(fā)出者與響應(yīng)者可以完全不認(rèn)識(shí)还栓,例如你在某個(gè)類中注冊(cè)了 A 觀察者某 E 事件的響應(yīng)歌殃,你可以在程序的任何代碼 X 中激發(fā) E,A 的相應(yīng)選擇器即被觸發(fā)蝙云,對(duì)象 A 與 X 完全是松散的氓皱。
最后,你的觀察者如果對(duì)一些事件沒興趣了勃刨,應(yīng)該從 NotificationCenter 中移除掉:
[[NSNotificationCenter defaultCenter] removeObserver:self name:@"NOTIFICATION_NAME" object:test];//object 與注冊(cè)時(shí)相同
//或[[NSNotificationCenter defaultCenter] removeObserver:self];
系統(tǒng)里定義了許多的 XxxNotification 名稱波材,其實(shí)只要 Cmd+Shift+O 打開 Open Quickly,輸入 nsnotification 或者 uinotification 可以看到許多以 Notification 結(jié)尾的變量定義身隐,由變量名稱也能理解在什么時(shí)候會(huì)激發(fā)什么事件廷区,一般都是向 [NSNotificationCenter defaultCenter] 通知的。
比如你想對(duì)系統(tǒng)的某些事件時(shí)作出響應(yīng)只要注冊(cè)一個(gè)觀察者即可贾铝,想要在每次鍵盤顯示后得到通知就得關(guān)心 UIKeyboardDidShowNotification 事件:
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(keyboardDidShow:) name:UIKeyboardDidShowNotification object:nil];
鍵盤顯示后就會(huì)執(zhí)行 self 的 keyboardDidShow 方法隙轻。
我們可以多看看第三方 Objective-C 庫是怎么運(yùn)用的 NSNotificationCenter。
Cocoa 給我們另一種事件通知模型就是 KVO(Key-Value Obsering)垢揩,基于 NSKeyValueObserving 非正式協(xié)議玖绿。
參考:
1.NSNotification Class Reference
3.iPhone之NSNotificationCenter使用方法