iOS-细数Objective-C中的回调机制

一、协议
协议主要是提供接口、或是类似C++多重继承功能,为类提供一种修饰机制。协议不是为回调而生的,它应该表述一组互操作约定。
优点:
实现简单,容易理解。
强类型检查。

缺点:
类与类间建立了比较强的耦合关系
有可能需要较长期保存委托以进行回调。如果保留的委托需要有独占性,可能会给单件模式、以及多线程带来麻烦。
类只能通过一个方法完成一种类型的回调。代码逻辑很容易集中到一个方法中。
大部分回调使用实际无需通过协议暴露给外部。

二、使用respondsToSelector和performSelector进行回调。
利用OBJC的运行时特性,查找对象的消息进行回调
优点:
与OBJC代码兼容性好。
具有延迟执行等特性。
轻量级的回调机制。

缺点:
回调产生的返回值只能为id类型,int等类型会产生错误。
参数最多只能传入两个。但可以通过建立包含多个参数的参数类进行回避。同时返回值限制也可通过此方式解决,即建立一个输入类和一个输出类。NSInvocation也提供了多参数的解决方法。

如果以
[target performSelector: @selector(callback)];
方式建立回调,则需要对类的回调消息名建立约定,且回调消息名具有独占性,即一个类中只能以此消息名进行回调。
如果通过外部传入SEL建立回调
[target performSelector: sel];
或是外部传入字符串建立回调
[target performSelector:NSSelectorFromString(@"callback")];
使用自动引数编译器特征(ARC)会产生警告“performSelector may cause a leak because its selector is unknown”
使用此种方式建立回调,当传入一个不符合约定的消息时,会产生副作用继续运行,而非报错。比如约定消息有2个参数,但传入消息只有1个参数,则按照参数约定顺序屏蔽掉最后传入的参数。或是传入消息具有3个参数,则多余的参数值未初始化。

三、函数指针
传统的C语言回调机制。
优点:
轻量级的回调机制。
只约定返回值和参数,而非函数名。无参数、返回值限制,使用灵活。
编译器提供类型检查。(错误时产生警告)

缺点:
与OBJC的消息机制不兼容。因为消息并非C语言中那样,函数名对应函数指针。即只能对C函数进行回调。
传入不符合约定的函数指针时,产生副作用继续运行,而非报错。

四、objc_msgSend
通过导入#import <objc/message.h>获得运行时的消息调用。
其定义为
id objc_msgSend(id theReceiver, SEL theSelector, ...)
优点:
轻量级的回调机制。
无传入参数限制。
相比performSelector,使用自动引数特征时,不产生警告。
同系列的方法支持double、struct等类型的返回值,但仍然不支持int型返回值(可使用NSNumber包装以回避)。

缺点:
传入不符合约定的消息时,产生副作用继续运行,而非报错。

五、IMP
IMP类似于OBJC提供的函数指针,它通过methodForSelector方法查询传入的Selector,以获得函数的入口地址。
其定义为
id (*IMP)(id, SEL, ...)
相比普通C语言的函数指针,其定义多了id,SEL这两个强制参数约定,其他与函数指针无异。
优点:
轻量级的回调机制。
传入不符合约定的消息时,报错。
无传入参数限制。返回值可通过强转获得,无类型限制。如:

      typedef int (*CBFUNC)(id, SEL, int, int, int); // 定义函数指针类型
      int ret = ((CBFUNC)callback)(self, sel, param1, param2, param3); // 强制转换

这里的id和SEL只是OBJC系统约定的占位,自定义回调时无实际意义。
由于此阶段实际是函数指针调用,因此最好还是typedef定义函数指针,然后对IMP强转一下,以免出现错误,也能提供一些编译期保护。

缺点:
依然不能提供如同协议和函数指针的编译期类型检查

六、NSNotificationCenter
NSNotificationCenter是OBJC提供的消息机制。它有些类似于观察者模式,通过关注感兴趣的消息,建立回调。NSNotificationCenter提供了一种低耦合的对象通讯机制,特别适合无指定对象的一对多回调。
主要方法:
1)获取消息中心实例(系统已创建,单件模式)
NSNotificationCenter *nc = [NSNotificationCenter defaultCenter];
2)发送消息。(事件发生时调用)

    NSNotificationCenter *nc = [NSNotificationCenter defaultCenter];
    [nc postNotificationName: NOTIFY_MSG_UC_COMMON_PLAYER_PLAY   // 消息名(字符串)
                      object:self                                // 消息源
                    userInfo:nil];                               // 用户字典(传递更多自定义参数)

3)注册消息

    [nc addObserver: self                              // 观察者
           selector: @selector(handleNotify_Play:)     // 回调
               name: NOTIFY_MSG_UC_COMMON_PLAYER_PLAY  // 监听消息
             object: nil];                             // 消息源
```
4)注销消息
```
    [nc removeObserver: self];
```
5)回调定义
```
    - (void) handleNotify_Play:(NSNotification *)note;
```
   只有一个参数
     NSNotification*
     –name      // 消息名
     –object    // 消息源
     –userInfo  // 用户字典

优点:
回调对象间耦合度低。相互之间可不必知道对方存在。
通过消息传递的信息无限制。
观察者可选择特定消息、特定对象,或者特定对象的特定消息进行观察。

缺点:
缺乏时序性。当事件发生时,回调执行的先后次序不确定。也不能等待回调完成执行后续操作。解决:1)使用传统回调机制。2)多线程时,可使用NSCondition同步线程。3)使用更多的消息。(过多使用可能导致混乱)

**七、Block**
Block是OBJC提供的一种运行时方法机制,类似于Javascript的匿名函数。它提供了一种运行时的临时回调机制。
Block对象的声明:
  声明一个参数为int,返回值为int的Block对象cb。
    int (^cb)(int);
  也可以通过typedef简化定义。
    typedef int(^BLOCK_CALLBACK_FUNC)(int);
    BLOCK_CALLBACK_FUNC cb = …
  回调函数定义:
```
    -(int)handleBlockCallbackFunc: (BLOCK_CALLBACK_FUNC)callback
    {
      return callback(10);
    }
```
回调函数使用:
  1)赋值后使用
```
     BLOCK_CALLBACK_FUNC cb =
       ^(int param)
       {
         NSLog(@"Block Msg: %d", param);
         return param*2;
       };
       int ret = [self handleBlockCallbackFunc:cb];
```
  2)使用时赋值
```
     int ret = [self handleBlockCallbackFunc:
                 ^(int param) {
                   NSLog(@"Block Msg: %d", param);
                   return param*2;
                 }];
```
注意:
**1)block对象使用的变量、参数在运行时被绑定,因此可以直接使用栈空间建立的变量,无需参数传入。但block对象的创建依然有生命周期限制,因此传入异步调用的block对象时,如果是栈空间创建的block,必须**
**使用Block_copy()将block拷出备份,然后使用Block_release()将block释放。参见Using Blocks章,Patterns to Avoid节**
**2)对于在栈空间声明的变量,绑定到block时被标记为const。只能读取不能写入。如果需要写入,需要用__block对变量进行标记。此时block使用的是从栈拷贝到堆中的对象。当出block时,如果栈可用则将堆中对象自动拷贝回栈。**
优点:
最轻量级的回调机制。
编译器类型检查。
如函数指针一样,灵活定义回调函数。

缺点:
执行效率。(影响程度不清楚)
容易导致代码逻辑集中。
IOS4之后的特性

**总结:**
OBJC还没有太完美的轻量级回调机制,只能根据情况选择合适的机制。

单纯的回调,且没有复用的必要,也无IOS版本限制,可采用block。
单纯的回调,有复用要求,可使用performSelector、objc_msgSend,或是IMP的回调机制。
使用自动引数的情况下,尽量不使用performSelector回调传入的@Selector,防止警告。
对象间有较多的互操作,对象有复用的必要,可采用协议。
无指定对象的一对多回调采用NSNotificationCenter。
有延迟调用等特殊应用的,可以使用performSelector。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,294评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,493评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,790评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,595评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,718评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,906评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,053评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,797评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,250评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,570评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,711评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,388评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,018评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,796评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,023评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,461评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,595评论 2 350

推荐阅读更多精彩内容