iOS 四种回调方法总结

最近对做IOS 项目遇到回调,抽空把相关资料整理下,以下是整理内容:

回调

回调就是将一段可执行的代码和一个特定的事件绑定起来。当特定的事件发生时,就会执行这段代码。
在Objective-C中,有四条途径可以实现回调。

目标-动作对

在程序开始定等待前,要求“当时间发生时,向指定的对象发送某个特定的信息”。这里接收消息的对象是目标,消息的选择器是动作。

辅助对象

在程序开始等待之前,要求“当时间发生时,向遵守相应协议的辅助对象发送消息”。委托对象和数据源是常见的辅助对象。

通知

苹果公司提供了一种称为通知中心的对象。在程序开始等待前,就可以告知通知中心”某个对象正在等待某些特定的通知。当其中的某个通知出现时,向指定的对象发送特定的消息”。当事件发生时,相关的对象会向通知中心发布通知,然后再由通知中心将通知转发给正在等待通知的对象。

Block对象

Block是一段可执行代码。在程序开始等待前,声明一个Block对象,当事件发生时,执行这段Block对象。

NSRunLoop

iOS中有一个NSRunLoop类,NSRunLoop实例会持续等待着,当特定的事件发生时,就会向相应的对象发送消息。NSRunLoop实例会在特定的事件发生时触发回调。

循环

实现回调之前要先创建一个循环:

int main(int argc, const char * argv[]) {
  @autoreleasepool {
    [[NSRunLoop currentRunLoop]run];
  }
  return 0;
}

目标-动作对

创建一个拥有NSRunLoop对象和NSTimer对象的应用程序。每隔两秒,NSTimer对象会向其目标发送指定的动作消息,创建一个新的类,名为BNRLogger,为NSTimer对象的目标。
在BNRLogger.h中声明动作方法:

#import <Foundation/Foundation.h>

@interface BNRLogger : NSObject<NSURLSessionDataDelegate>
@property(nonatomic) NSDate *lastTime;

-(NSString *) lastTimeString;
-(void)updateLastTime: (NSTimer *) t;

@end

在BNRLogger.m中实现方法:

#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中创建一个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已经写过NSURLSession方法的使用,那么辅助对象回调的使用,将BNRLogger对象成为NSURLSession的委托对象,特定的事件发生时,该对象会向辅助对象发送消息。
main.m中创建一个NSURL对象以及NSURLRequest对象。然后创建一个NSURLSession对象,设置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.根据会话对象创建一个Task(发送请求)
    NSURLSessionDataTask *dataTask = [session dataTaskWithRequest:request];

    //5.执行任务
    [dataTask resume];

    [[NSRunLoop currentRunLoop]run];

  }
  return 0;
}

BNRLogger.h中,声明NSURLSessionDataDelegate协议:

#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.接收到服务器响应的时候调用该方法
-(void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveResponse:(NSURLResponse *)response completionHandler:(void (^)(NSURLSessionResponseDisposition))completionHandler
{
  //在该方法中可以得到响应头信息,即response
  NSLog(@"didReceiveResponse--%@",[NSThread currentThread]);
  NSLog(@"响应");
  //注意:需要使用completionHandler回调告诉系统应该如何处理服务器返回的数据
  //默认是取消的
  /*
   NSURLSessionResponseCancel = 0,    默认的处理方式,取消
   NSURLSessionResponseAllow = 1,     接收服务器返回的数据
   NSURLSessionResponseBecomeDownload = 2,变成一个下载请求
   NSURLSessionResponseBecomeStream    变成一个流
   */

  completionHandler(NSURLSessionResponseAllow);
}

//2.接收到服务器返回数据的时候会调用该方法,如果数据较大那么该方法可能会调用多次
-(void)URLSession:(NSURLSession *)session dataTask:(NSURLSessionDataTask *)dataTask didReceiveData:(NSData *)data
{
  NSLog(@"didReceiveData--%@",[NSThread currentThread]);
  NSLog(@"返回");
  //拼接服务器返回的数据
  [self.responseData appendData:data];
}

//3.当请求完成(成功|失败)的时候会调用该方法,如果请求失败,则error有值
-(void)URLSession:(NSURLSession *)session task:(NSURLSessionTask *)task didCompleteWithError:(NSError *)error
{
  NSLog(@"didCompleteWithError--%@",[NSThread currentThread]);
  NSLog(@"完成");
  if(error == nil)
  {
    //解析数据
    NSDictionary *dict = [NSJSONSerialization JSONObjectWithData:self.responseData options:kNilOptions error:nil];
    NSLog(@"%@",dict);
  }
}

@end

通知

当系统时区发生变化时,会向通知中心发布NSSystemTimeZoneDidChangeNotification通知,然后通知中心会将该通知转发给相应的观察者。

main.m中将BNRLogger实例注册为观察者,系统时区设置发生变化可以收到相应的通知:

 //在”辅助对象”方法应用程序中的main.m中加入这行代码
 [[NSNotificationCenter defaultCenter]addObserver:logger selector:@selector(zoneChange:) name:NSSystemTimeZoneDidChangeNotification object:nil];

在BNRLogger.m中实现该方法:

 //在”辅助对象”方法应用程序中的BNRLogger.m中加入这行代码
 -(void)zoneChange:(NSNotification *)note
{
  NSLog(@"The system time zone has changed!");
}

Block回调

把上面所讲的“通知”方法应用程序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!");
    }];

“通知”方法应用程序BNRLogger.m中的这个方法去掉:

-(void)zoneChange:(NSNotification *)note
{
  NSLog(@"The system time zone has changed!");
}

总结

  1. 对于只做一件事情的对象(例如),使用目标-动作对。
  2. 对于功能更复杂的对象(例如NSURLSession),使用辅助对象。最常见的辅助对象类型是委托对象。
  3. 对于要触发多个(其他对象中的)回调的对象(例如NSTimeZone),使用通知。
  4. Block实现回调使代码便于阅读。

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

(0)

相关推荐

  • 详解优化iOS程序性能的25个方法

    1. 用ARC管理内存 ARC(Automatic ReferenceCounting, 自动引用计数)和iOS5一起发布,它避免了最常见的也就是经常是由于我们忘记释放内存所造成的内存泄露.它自动为你管理retain和release的过程,所以你就不必去手动干预了.忘掉代码段结尾的release简直像记得吃饭一样简单.而ARC会自动在底层为你做这些工作.除了帮你避免内存泄露,ARC还可以帮你提高性能,它能保证释放掉不再需要的对象的内存. 现在所有的iOS程序都用ARC了,这条可以忽略. 2. 在

  • iOS开发 全机型适配解决方法

    最近做项目,对于IPhone 手机机型适配很是头疼,所以整理下网上资料,记录下来,也许能帮助到正看文章的你, 今天打算跟大家聊聊最近研究的全机型适配思路. 当前我们需要适配的iPhone机型有4s.5s.6s.6Plus四种机型.它们的尺寸分别是 iphone4s {320, 480}                           960*640  iphone5 5s {320, 568}                       1136*640  iphone6 6s   {375

  • 详解IOS点击空白处隐藏键盘的几种方法介绍

    IOS7 点击空白处隐藏键盘的几种方法,具体如下: iOS开发中经常要用到输入框,默认情况下点击输入框就会弹出键盘,但是必须要实现输入框return的委托方法才能取消键盘的显示,对于用户体验来说很不友好,我们可以实现点击键盘以外的空白区域来将键盘隐藏,以下我总结出了几种隐藏键盘的方法: 首先说明两种可以让键盘隐藏的Method: 1.[view endEditing:YES]  这个方法可以让整个view取消第一响应者,从而让所有控件的键盘隐藏. 2.[textFiled resignFirst

  • iOS点击推送消息跳到应用指定页面方法

    现在的推送用的越来越频繁,几乎每个应用都开始用到了.其实又有几个用户会去看推送消息呢?没办法,产品经理最大啊,只是苦了我们这一帮程序员啊!闲话少说,进入正题.兄弟我用的是极光推送,自然是以极光推送为例了. 现在点击推送消息,有两种跳转方式:1.打开应用,跳转到应用首页:2.打开应用,跳转到指定页面. ​第一种,你什么都不用设置,只要注册极光应用就可以.这里就不写怎么注册极光应用了,可以参考官方文档,写的很详细. 第二种,重头戏来了. // APP未运行时获取通知的内容 remoteNotific

  • iOS 防止按钮多次点击造成多次响应的方法

    iOS 防止按钮多次点击造成多次响应的方法 在日常开发中经常会碰到一种bug就是因为用户快速点击某个按钮,导致页面重复push或者重复发送网络请求.这样的问题既对用户体验有影响,而且还会一定程度上增加服务器的压力. 目前,我为了防止按钮快速点击主要使用以下两种办法 1.在每次点击时先取消之前的操作(网上看到的方法) - (void)buttonClicked:(id)sender { //这里是关键,点击按钮后先取消之前的操作,再进行需要进行的操作 [[self class] cancelPre

  • iOS拍照后图片自动旋转90度的完美解决方法

    今天开发一个拍照获取照片的功能的时候, 发现上传之后图片会自动旋转90. 测试发现, 只要是图片大于2M, 系统就会自动翻转照片 相机拍照后直接取出来的UIimage(用UIImagePickerControllerOriginalImage取出),它本身的imageOrientation属性是3,即UIImageOrientationRight.如果这个图片直接使用则没事,但是如果对它进行裁剪.缩放等操作后,它的这个imageOrientation属性会变成0.此时这张图片用在别的地方就会发生

  • iOS常用的公共方法详解

    1. 获取磁盘总空间大小 //磁盘总空间 + (CGFloat)diskOfAllSizeMBytes{ CGFloat size = 0.0; NSError *error; NSDictionary *dic = [[NSFileManager defaultManager] attributesOfFileSystemForPath:NSHomeDirectory() error:&error]; if (error) { #ifdef DEBUG NSLog(@"error: %

  • iOS中生成指定大小、指定颜色的二维码和条形码方法详解

    iOS7.0之后可以利用系统原生 API 生成二维码, iOS8.0之后可以生成条形码, 系统默认生成的颜色是黑色. 在这里, 利用以下方法可以生成指定大小.指定颜色的二维码和条形码, 还可以添加背景颜色.阴影效果, 以下是具体方法. 一. 生成二维码 Avilable in iOS 7.0 and later 方法如下: #pragma mark - 生成二维码 //Avilable in iOS 7.0 and later + (UIImage *)qrCodeImageWithConten

  • iOS文字渐变色效果的实现方法

    照例先上文字渐变的效果图 实现思路如下 一.创建一个颜色渐变层,渐变图层跟文字控件一样大. 二.用文字图层裁剪渐变层,只保留文字部分,就会让渐变层只保留有文字的部分,相当于间接让渐变层显示文字,我们看到的其实是被裁剪过后,渐变层的部分内容. 注意:如果用文字图层裁剪渐变层,文字图层就不在拥有显示功能,这个图层就被弄来裁剪了,不会显示,在下面代码中也会有说明. 2.1 创建一个带有文字的label,label能显示文字. 2.2 设置渐变图层的mask为label图层,就能用文字裁剪渐变图层了.

  • IOS开发中键盘输入屏幕上移的解决方法

    在IOS开法中经常会遇到键盘遮挡屏幕的事情(比如输入账号密码验证码等等),就使得原本都不大的屏幕直接占了一半甚至更多的位置,这倒无所谓,关键是挡住了下面的按钮.这样的话按钮的事件也就触发不了,最好的解决办法就是当输入这些信息的时候让整个屏幕上移一个键盘的位置,或者上移到指定的位置. 首先一般输入的话都用的是UITextField,所以要监听用户什么时候开始输入和什么时候结束输入,直接设置代理代理就行了,要遵受 UITextFieldDelegate协议. //遵循协议 @interface Vi

随机推荐