iOS 四種回調(diào)方法總結(jié)
最近對做IOS 項目遇到回調(diào),抽空把相關(guān)資料整理下,以下是整理內(nèi)容:
回調(diào)
回調(diào)就是將一段可執(zhí)行的代碼和一個特定的事件綁定起來。當(dāng)特定的事件發(fā)生時,就會執(zhí)行這段代碼。
在Objective-C中,有四條途徑可以實現(xiàn)回調(diào)。
目標(biāo)-動作對
在程序開始定等待前,要求“當(dāng)時間發(fā)生時,向指定的對象發(fā)送某個特定的信息”。這里接收消息的對象是目標(biāo),消息的選擇器是動作。
輔助對象
在程序開始等待之前,要求“當(dāng)時間發(fā)生時,向遵守相應(yīng)協(xié)議的輔助對象發(fā)送消息”。委托對象和數(shù)據(jù)源是常見的輔助對象。
通知
蘋果公司提供了一種稱為通知中心的對象。在程序開始等待前,就可以告知通知中心”某個對象正在等待某些特定的通知。當(dāng)其中的某個通知出現(xiàn)時,向指定的對象發(fā)送特定的消息”。當(dāng)事件發(fā)生時,相關(guān)的對象會向通知中心發(fā)布通知,然后再由通知中心將通知轉(zhuǎn)發(fā)給正在等待通知的對象。
Block對象
Block是一段可執(zhí)行代碼。在程序開始等待前,聲明一個Block對象,當(dāng)事件發(fā)生時,執(zhí)行這段Block對象。
NSRunLoop
iOS中有一個NSRunLoop類,NSRunLoop實例會持續(xù)等待著,當(dāng)特定的事件發(fā)生時,就會向相應(yīng)的對象發(fā)送消息。NSRunLoop實例會在特定的事件發(fā)生時觸發(fā)回調(diào)。
循環(huán)
實現(xiàn)回調(diào)之前要先創(chuàng)建一個循環(huán):
int main(int argc, const char * argv[]) { @autoreleasepool { [[NSRunLoop currentRunLoop]run]; } return 0; }
目標(biāo)-動作對
創(chuàng)建一個擁有NSRunLoop對象和NSTimer對象的應(yīng)用程序。每隔兩秒,NSTimer對象會向其目標(biāo)發(fā)送指定的動作消息,創(chuàng)建一個新的類,名為BNRLogger,為NSTimer對象的目標(biāo)。
在BNRLogger.h中聲明動作方法:
#import <Foundation/Foundation.h> @interface BNRLogger : NSObject<NSURLSessionDataDelegate> @property(nonatomic) NSDate *lastTime; -(NSString *) lastTimeString; -(void)updateLastTime: (NSTimer *) t; @end
在BNRLogger.m中實現(xiàn)方法:
#import "BNRLogger.h" @implementation BNRLogger -(NSString *)lastTimeString { static NSDateFormatter *dateFormatter=nil; if(!dateFormatter) { dateFormatter =[[NSDateFormatter alloc]init]; [dateFormatter setTimeStyle:NSDateFormatterMediumStyle]; [dateFormatter setDateStyle:NSDateFormatterMediumStyle]; NSLog(@"created dateFormatter"); } return [dateFormatter stringFromDate:self.lastTime]; } -(void)updateLastTime:(NSTimer *)t { NSDate *now=[NSDate date]; [self setLastTime:now]; NSLog(@"Just set time to %@",self.lastTimeString); } @end
main.m中創(chuàng)建一個BNRLogger實例:
#import <Foundation/Foundation.h> #import "BNRLogger.h" int main(int argc, const char * argv[]) { @autoreleasepool { BNRLogger *logger=[[BNRLogger alloc]init]; __unused NSTimer *timer=[NSTimer scheduledTimerWithTimeInterval:2.0 target:logger selector:@selector(updateLastTime:) userInfo:nil repeats:YES]; [[NSRunLoop currentRunLoop]run]; } return 0; }
輔助對象
我的上一篇Blog已經(jīng)寫過NSURLSession方法的使用,那么輔助對象回調(diào)的使用,將BNRLogger對象成為NSURLSession的委托對象,特定的事件發(fā)生時,該對象會向輔助對象發(fā)送消息。
main.m中創(chuàng)建一個NSURL對象以及NSURLRequest對象。然后創(chuàng)建一個NSURLSession對象,設(shè)置BNRLogger的實例為它的
委托對象:
#import <Foundation/Foundation.h> #import "BNRLogger.h" int main(int argc, const char * argv[]) { @autoreleasepool { BNRLogger *logger=[[BNRLogger alloc]init]; //URL是一張圖片的下載鏈接 NSURL *url = [NSURL URLWithString:@"http://image.baidu.com/search/down?tn=download&ipn=dwnl&word=download&ie=utf8&fr=result&url=http%3A%2F%2Fimg.xiazaizhijia.com%2Fuploads%2F2016%2F0914%2F20160914112151862.jpg&thumburl=http%3A%2F%2Fimg4.imgtn.bdimg.com%2Fit%2Fu%3D2349180720%2C2436282788%26fm%3D11%26gp%3D0.jpg"]; NSURLRequest *request = [NSURLRequest requestWithURL:url]; __unused NSURLSession *session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration] delegate:logger delegateQueue:[NSOperationQueue mainQueue]]; __unused NSTimer *timer=[NSTimer scheduledTimerWithTimeInterval:2.0 target:logger selector:@selector(updateLastTime:) userInfo:nil repeats:YES]; //4.根據(jù)會話對象創(chuàng)建一個Task(發(fā)送請求) NSURLSessionDataTask *dataTask = [session dataTaskWithRequest:request]; //5.執(zhí)行任務(wù) [dataTask resume]; [[NSRunLoop currentRunLoop]run]; } return 0; }
BNRLogger.h中,聲明NSURLSessionDataDelegate協(xié)議:
#import <Foundation/Foundation.h> @interface BNRLogger : NSObject<NSURLSessionDataDelegate> @property (nonatomic, strong) NSMutableData *responseData; @property(nonatomic) NSDate *lastTime; -(NSString *) lastTimeString; -(void)updateLastTime: (NSTimer *) t; @end
BNRLogger.m中,有NSURLSession的代理方法,具體可以看NSURLSession:
#import "BNRLogger.h" @implementation BNRLogger -(NSMutableData *)responseData { if (_responseData == nil) { _responseData = [NSMutableData data]; } return _responseData; } -(NSString *)lastTimeString { static NSDateFormatter *dateFormatter=nil; if(!dateFormatter) { dateFormatter =[[NSDateFormatter alloc]init]; [dateFormatter setTimeStyle:NSDateFormatterMediumStyle]; [dateFormatter setDateStyle:NSDateFormatterMediumStyle]; NSLog(@"created dateFormatter"); } return [dateFormatter stringFromDate:self.lastTime]; } -(void)updateLastTime:(NSTimer *)t { NSDate *now=[NSDate date]; [self setLastTime:now]; NSLog(@"Just set time to %@",self.lastTimeString); } //1.接收到服務(wù)器響應(yīng)的時候調(diào)用該方法 -(void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler { //在該方法中可以得到響應(yīng)頭信息,即response NSLog(@"didReceiveResponse--%@",[NSThread currentThread]); NSLog(@"響應(yīng)"); //注意:需要使用completionHandler回調(diào)告訴系統(tǒng)應(yīng)該如何處理服務(wù)器返回的數(shù)據(jù) //默認(rèn)是取消的 /* NSURLSessionResponseCancel = 0, 默認(rèn)的處理方式,取消 NSURLSessionResponseAllow = 1, 接收服務(wù)器返回的數(shù)據(jù) NSURLSessionResponseBecomeDownload = 2,變成一個下載請求 NSURLSessionResponseBecomeStream 變成一個流 */ completionHandler(NSURLSessionResponseAllow); } //2.接收到服務(wù)器返回數(shù)據(jù)的時候會調(diào)用該方法,如果數(shù)據(jù)較大那么該方法可能會調(diào)用多次 -(void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data { NSLog(@"didReceiveData--%@",[NSThread currentThread]); NSLog(@"返回"); //拼接服務(wù)器返回的數(shù)據(jù) [self.responseData appendData:data]; } //3.當(dāng)請求完成(成功|失敗)的時候會調(diào)用該方法,如果請求失敗,則error有值 -(void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error { NSLog(@"didCompleteWithError--%@",[NSThread currentThread]); NSLog(@"完成"); if(error == nil) { //解析數(shù)據(jù) NSDictionary *dict = [NSJSONSerialization JSONObjectWithData:self.responseData options:kNilOptions error:nil]; NSLog(@"%@",dict); } } @end
通知
當(dāng)系統(tǒng)時區(qū)發(fā)生變化時,會向通知中心發(fā)布NSSystemTimeZoneDidChangeNotification通知,然后通知中心會將該通知轉(zhuǎn)發(fā)給相應(yīng)的觀察者。
main.m中將BNRLogger實例注冊為觀察者,系統(tǒng)時區(qū)設(shè)置發(fā)生變化可以收到相應(yīng)的通知:
//在”輔助對象”方法應(yīng)用程序中的main.m中加入這行代碼 [[NSNotificationCenter defaultCenter]addObserver:logger selector:@selector(zoneChange:) name:NSSystemTimeZoneDidChangeNotification object:nil];
在BNRLogger.m中實現(xiàn)該方法:
//在”輔助對象”方法應(yīng)用程序中的BNRLogger.m中加入這行代碼 -(void)zoneChange:(NSNotification *)note { NSLog(@"The system time zone has changed!"); }
Block回調(diào)
把上面所講的“通知”方法應(yīng)用程序main.m中的:
[[NSNotificationCenter defaultCenter]addObserver:logger selector:@selector(zoneChange:) name:NSSystemTimeZoneDidChangeNotification object:nil];
改為:
[[NSNotificationCenter defaultCenter]addObserverForName:NSSystemTimeZoneDidChangeNotification object:nil queue:[NSOperationQueue mainQueue] usingBlock:^(NSNotification *note){ NSLog(@"The system time zone has changed!"); }];
“通知”方法應(yīng)用程序BNRLogger.m中的這個方法去掉:
-(void)zoneChange:(NSNotification *)note { NSLog(@"The system time zone has changed!"); }
總結(jié)
- 對于只做一件事情的對象(例如),使用目標(biāo)-動作對。
- 對于功能更復(fù)雜的對象(例如NSURLSession),使用輔助對象。最常見的輔助對象類型是委托對象。
- 對于要觸發(fā)多個(其他對象中的)回調(diào)的對象(例如NSTimeZone),使用通知。
- Block實現(xiàn)回調(diào)使代碼便于閱讀。
感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
相關(guān)文章
Flutter繪制3.4邊形及多邊形漸變動畫實現(xiàn)示例
這篇文章主要為大家介紹了Flutter繪制3.4邊形之多邊形漸變動畫實現(xiàn)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-08-08iOS開發(fā)教程之APP內(nèi)部切換語言的實現(xiàn)方法
這篇文章主要給大家介紹了關(guān)于iOS開發(fā)教程之APP內(nèi)部切換語言的實現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧2019-02-02解決iOS11圖片下拉放大出現(xiàn)信號欄白條的bug問題
這篇文章主要介紹了iOS11圖片下拉放大出現(xiàn)信號欄白條的bug問題,需要的朋友參考下吧2017-09-09iOS App中實現(xiàn)播放音效和音樂功能的簡單示例
這篇文章主要介紹了iOS App中實現(xiàn)播放音效和音樂功能的簡單示例,示例代碼為傳統(tǒng)的Objective-C,需要的朋友可以參考下2016-03-03iOS Swift 值類型與引用類型使用區(qū)別基礎(chǔ)詳解
這篇文章主要為大家介紹了iOS Swift 值類型與引用類型使用區(qū)別基礎(chǔ)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-07-07