一、NSNotification和Delegate的聯(lián)系和區(qū)別
眾所周知刊头,IOS中經(jīng)常會使用到NSNotification和delegate來進(jìn)行一些類之間的消息傳遞眠寿。言歸正傳莽鸿,這兩種有什么區(qū)別呢?
NSNotification就是IOS提供的一個消息中心拜轨,由一個全局的defaultNotification管理應(yīng)用中的消息機(jī)制抽减。通過公開的API可以看出,這里面使用了是一個觀察者橄碾,通過注冊addObserver和解除注冊removeObserver來實(shí)現(xiàn)消息傳遞卵沉。蘋果文檔特別提出,在類析構(gòu)的時候法牲,要記得把removeObserver史汗,不然就會引發(fā)崩潰,所以NSNotifcation的使用是沒有retain+1的拒垃,NSNotification是一對多的停撞。
至于Delegate,很簡單悼瓮,就是通過增加一個指針戈毒,然后把需要調(diào)用的函數(shù)通過delegate傳遞到其他類中,來得很直截了當(dāng)横堡。不需要通過廣播的形式去實(shí)現(xiàn)埋市,但是,delegate的形式只能是一對一命贴,不能實(shí)現(xiàn)一對多恐疲。
在什么情況下使用Delegate和NSNotifiation呢?
從效率上看Delegate是一個很輕量級的腊满,相對delegate套么,NSNotification卻是一個很重量級的培己,效率上delegate明顯要比Noticication高。一般情況我們會這樣使用胚泌。
場景一:
A擁有B省咨,然后B中的一些操作需要回調(diào)到A中,這時候就簡單的通過delegate回調(diào)到A玷室。因?yàn)锽是A創(chuàng)建的零蓉,B可以很直接的把delegate賦值A(chǔ)。
場景二:
A和B是兩個不相干的關(guān)系穷缤,A不知道B敌蜂,B也不知道A,那么這時候如果通過delegate就沒辦法做到津肛,會相對復(fù)雜章喉。所以可以通過NSNotifcation去做一些消息傳遞。
所以使用delegate的情況是兩者有直接的關(guān)系身坐,至于一方知道另一方的存在秸脱。而NSNotifcation一般是大家不知道對方的存在,一般是使用跨模塊的時候使用部蛇。在使用的時候摊唇,使用delegate可能需要多寫一些delegate去實(shí)現(xiàn),代碼量比較多涯鲁。NSNotication只要定義相關(guān)的NotificationName就可以很方便的溝通巷查。兩者各有所長。
二抹腿、監(jiān)聽系統(tǒng)自帶的NSNotification
系統(tǒng)里定義了許多的 XxxNotification 名稱岛请,其實(shí)只要 Cmd+Shift+O 打開 Open Quickly,輸入 NSNotification 或者 UINotification 可以看到許多以 Notification 結(jié)尾的變量定義幢踏,由變量名稱也能理解在什么時候會激發(fā)什么事件髓需,一般都是向 [NSNotificationCenter defaultCenter] 通知的。
使用步驟
第一步:注冊系統(tǒng)監(jiān)聽事件
//在NSNotificationCenter中注冊鍵盤彈出事件
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(keyboardUpEvent:) name:UIKeyboardDidShowNotification object:nil];
//在NSNotificationCenter中注冊鍵盤隱藏事件
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(keyboardDownEvent:) name:UIKeyboardDidHideNotification object:nil];
//在NSNotificationCenter中注冊程序從后臺喚醒事件
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(becomeActive:) name:UIApplicationDidBecomeActiveNotification object:nil];
第二步:事件觸發(fā)后的處理
/**
* 彈出鍵盤事件觸發(fā)處理
*
* @param notification
*/
-(void)keyboardUpEvent : (NSNotification *)notification{
//NSLog(@"鍵盤彈出事件觸發(fā)==%@",notification);
NSLog(@"鍵盤彈出事件觸發(fā)");
}
/**
* 鍵盤隱藏事件觸發(fā)處理
*
* @param notification
*/
-(void)keyboardDownEvent : (NSNotification *)notification{
//NSLog(@"鍵盤隱藏事件觸發(fā)==%@",notification);
NSLog(@"鍵盤隱藏事件觸發(fā)");
}
/**
* 程序從后臺喚醒觸發(fā)處理
*
* @param notification
*/
-(void)becomeActive: (NSNotification *)notification{
NSLog(@"程序從后臺喚醒觸發(fā)處理");
}
第三步房蝉、在dealloc中解除監(jiān)聽
/**
*NSNotificationCenter 注意點(diǎn):每一次在接受者對象中需要delleac把它銷毀掉僚匆。
*/
-(void)dealloc{
[[NSNotificationCenter defaultCenter] removeObserver:self];
}
三、自定義NSNotification
這里我使用的一個實(shí)例為:在ViewController中定義一個按鈕搭幻,點(diǎn)擊該按鈕咧擂,同時改變兩個自定義View中的內(nèi)容。
使用步驟
第一步檀蹋、在ViewController中生成一個按鈕松申,兩個自定義View
UIButton *postMsgBtn = [[UIButton alloc] initWithFrame:CGRectMake(50, 200, 100, 40)];
[postMsgBtn setTitle:@"發(fā)送消息" forState:UIControlStateNormal];
postMsgBtn.backgroundColor = [UIColor grayColor];
[postMsgBtn addTarget:self action:@selector(postMsg:) forControlEvents:UIControlEventTouchUpInside];
[self.view addSubview:postMsgBtn];
MyView *view = [[MyView alloc] initWithFrame:CGRectMake(50, 250, 100, 50)];
[self.view addSubview:view];
MyView *view2 = [[MyView alloc] initWithFrame:CGRectMake(50, 320, 100, 50)];
[self.view addSubview:view2];
第二步、點(diǎn)擊按鈕,發(fā)送Notification
-(void)postMsg: (UIButton *)btn{
[[NSNotificationCenter defaultCenter] postNotificationName:NOTIFICATION_MESSAGE_NAME object:nil userInfo:@{@"msg":@"jingming1"}];
}
第三步贸桶、在自定義View中注冊監(jiān)聽事件
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(acceptMsg:) name:NOTIFICATION_MESSAGE_NAME object:nil];
第四步舅逸、處理監(jiān)聽事件
-(void)acceptMsg : (NSNotification *)notification{
NSLog(@"%@",notification);
NSDictionary *userInfo = notification.userInfo;
_label.text = [userInfo objectForKey:@"msg"];
}
第五步、在dealloc中解除監(jiān)聽
-(void)dealloc{
[[NSNotificationCenter defaultCenter] removeObserver:self];
}