最近对做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://img.xiazaizhijia.com/uploads/2016/0914/20160914112151862.jpg&thumburl=http://img4.imgtn.bdimg.com/it/u=2349180720,2436282788&fm=11&gp=0.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实现回调使代码便于阅读。

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

iOS 四种回调方法总结的更多相关文章

  1. HTML5在微信内置浏览器下右上角菜单的调整字体导致页面显示错乱的问题

    HTML5在微信内置浏览器下,在右上角菜单的调整字体导致页面显示错乱的问题,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧

  2. iOS实现拖拽View跟随手指浮动效果

    这篇文章主要为大家详细介绍了iOS实现拖拽View跟随手指浮动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  3. ios – containerURLForSecurityApplicationGroupIdentifier:在iPhone和Watch模拟器上给出不同的结果

    我使用默认的XCode模板创建了一个WatchKit应用程序.我向iOSTarget,WatchkitAppTarget和WatchkitAppExtensionTarget添加了应用程序组权利.(这是应用程序组名称:group.com.lombax.fiveminutes)然后,我尝试使用iOSApp和WatchKitExtension访问共享文件夹URL:延期:iOS应用:但是,测试NSURL

  4. ios – Testflight无法安装应用程序

    我有几个测试人员注册了testflight并连接了他们的设备……他们有不同的ios型号……但是所有这些都有同样的问题.当他们从“safari”或“testflight”应用程序本身单击应用程序的安装按钮时……达到约90%并出现错误消息…

  5. ibm-mobilefirst – 在iOS 7.1上获取“无法安装应用程序,因为证书无效”错误

    当我的客户端将他们的设备更新到iOS7.1,然后尝试从AppCenter更新我们的应用程序时,我收到了上述错误.经过一番搜索,我找到了一个类似问题的帖子here.但是后来因为我在客户端使用AppCenter更新应用程序的环境中,我无法使用USB插件并为他们安装应用程序.在发布支持之前,是否有通过AppCenter进行下载的解决方法?

  6. ios – 视图的简单拖放?

    我正在学习iOS,但我找不到如何向UIView添加拖放行为.我试过了:它说“UIView没有可见的接口声明选择器addTarget”此外,我尝试添加平移手势识别器,但不确定这是否是我需要的它被称为,但不知道如何获得事件的坐标.在iOS中注册移动事件回调/拖放操作的标准简单方法是什么?

  7. ios – 什么控制iTunes中iPhone应用程序支持的语言列表?

    什么控制iPhone应用程序的iTunes页面中支持的语言?

  8. ios – 获得APNs响应BadDeviceToken或Unregistered的可能原因是什么?

    我知道设备令牌在某些时候是有效的.用户如何使其设备令牌变坏?从关于“未注册”的文档:Thedevicetokenisinactiveforthespecifiedtopic.这是否意味着应用程序已被删除?.您应该看到四种分发方法:如果您选择AppStore或Enterprise,您将在后面的对话框中看到Xcode将APNS权利更改为生产:如果选择AdHoc或Development,则aps-environment下的文本将是开发,然后应与后端的配置匹配.

  9. ios – 当我关闭应用程序时,我从调试器获得消息:由于信号15而终止

    我怎么能解决这个问题,我不知道这个链接MypreviousproblemaboutCoredata对我的问题有影响吗?当我cmd应用程序的Q时,将出现此消息.Messagefromdebugger:Terminatedduetosignal15如果谁知道我以前的问题的解决方案,请告诉我.解决方法>来自调试器的消息:每当用户通过CMD-Q(退出)或STOP手动终止应用程序(无论是在iOS模拟器中还是

  10. ios – NSUbiquityIdentityDidChangeNotification和SIGKILL

    当应用程序被发送到后台时,我们会删除观察者吗?我遇到的问题是,当UbiquityToken发生变化时,应用程序终止,因为用户已经更改了iCloud设置.你们如何设法订阅这个通知,如果你不这样做,你会做什么来跟踪当前登录的iCloud用户?

随机推荐

  1. iOS实现拖拽View跟随手指浮动效果

    这篇文章主要为大家详细介绍了iOS实现拖拽View跟随手指浮动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  2. iOS – genstrings:无法连接到输出目录en.lproj

    使用我桌面上的项目文件夹,我启动终端输入:cd然后将我的项目文件夹拖到终端,它给了我路径.然后我将这行代码粘贴到终端中找.-name*.m|xargsgenstrings-oen.lproj我在终端中收到此错误消息:genstrings:无法连接到输出目录en.lproj它多次打印这行,然后说我的项目是一个目录的路径?没有.strings文件.对我做错了什么的想法?

  3. iOS 7 UIButtonBarItem图像没有色调

    如何确保按钮图标采用全局色调?解决方法只是想将其转换为根注释,以便为“回答”复选标记提供更好的上下文,并提供更好的格式.我能想出这个!

  4. ios – 在自定义相机层的AVFoundation中自动对焦和自动曝光

    为AVFoundation定制图层相机创建精确的自动对焦和曝光的最佳方法是什么?

  5. ios – Xcode找不到Alamofire,错误:没有这样的模块’Alamofire’

    我正在尝试按照github(https://github.com/Alamofire/Alamofire#cocoapods)指令将Alamofire包含在我的Swift项目中.我创建了一个新项目,导航到项目目录并运行此命令sudogeminstallcocoapods.然后我面临以下错误:搜索后我设法通过运行此命令安装cocoapodssudogeminstall-n/usr/local/bin

  6. ios – 在没有iPhone6s或更新的情况下测试ARKit

    我在决定下载Xcode9之前.我想玩新的框架–ARKit.我知道要用ARKit运行app我需要一个带有A9芯片或更新版本的设备.不幸的是我有一个较旧的.我的问题是已经下载了新Xcode的人.在我的情况下有可能运行ARKit应用程序吗?那个或其他任何模拟器?任何想法或我将不得不购买新设备?解决方法任何iOS11设备都可以使用ARKit,但是具有高质量AR体验的全球跟踪功能需要使用A9或更高版本处理器的设备.使用iOS11测试版更新您的设备是必要的.

  7. 将iOS应用移植到Android

    我们制作了一个具有2000个目标c类的退出大型iOS应用程序.我想知道有一个最佳实践指南将其移植到Android?此外,由于我们的应用程序大量使用UINavigation和UIView控制器,我想知道在Android上有类似的模型和实现.谢谢到目前为止,guenter解决方法老实说,我认为你正在计划的只是制作难以维护的糟糕代码.我意识到这听起来像很多工作,但从长远来看它会更容易,我只是将应用程序的概念“移植”到android并从头开始编写.

  8. ios – 在Swift中覆盖Objective C类方法

    我是Swift的初学者,我正在尝试在Swift项目中使用JSONModel.我想从JSONModel覆盖方法keyMapper,但我没有找到如何覆盖模型类中的Objective-C类方法.该方法的签名是:我怎样才能做到这一点?解决方法您可以像覆盖实例方法一样执行此操作,但使用class关键字除外:

  9. ios – 在WKWebView中获取链接URL

    我想在WKWebView中获取tapped链接的url.链接采用自定义格式,可触发应用中的某些操作.例如HTTP://我的网站/帮助#深层链接对讲.我这样使用KVO:这在第一次点击链接时效果很好.但是,如果我连续两次点击相同的链接,它将不报告链接点击.是否有解决方法来解决这个问题,以便我可以检测每个点击并获取链接?任何关于这个的指针都会很棒!解决方法像这样更改addobserver在observeValue函数中,您可以获得两个值

  10. ios – 在Swift的UIView中找到UILabel

    我正在尝试在我的UIViewControllers的超级视图中找到我的UILabels.这是我的代码:这是在Objective-C中推荐的方式,但是在Swift中我只得到UIViews和CALayer.我肯定在提供给这个方法的视图中有UILabel.我错过了什么?我的UIViewController中的调用:解决方法使用函数式编程概念可以更轻松地实现这一目标.

返回
顶部