Effective Objective-C 2.0 — 第12条:理解消息转发机制

11 条讲解了对象的消息传递机制

12条讲解对象在收到无法解读的消息之后会发生什么,就会启动“消息转发”(message forwarding)机制,

  • 若对象无法响应某个选择子,则进入消息转发流程。
  • 1,通过运行期的动态方法解析功能,可以在需要用到某个方法时再将其加入类中。
  • 2,对象可以把其无法解读的某些选择子转交给其他对象来处理。
  • 经过上述两步之后,如果还是没办法处理选择子,那就启动完整的消息转发机制。

动态方法解析

对象在收到无法解读的消息之后,首先调用其所属类的下列类方法

+ (BOOL)resolveInstanceMethod:(SEL)selector

返回值为BOOL,表示这个类是否能新增一个实例方法用以处理此选择子。

使用这种方法的前提是:相关方法的实现代码已经写好,只等着运行的时候动态插在类里面就可以了。此方案常用来实现@dynamic属性,比如说要访问CoreData框架中NSManagedObjects对象的属性时就可以这么做,因为实现这些属性所需的存取方法在编译期就能确定。

下列代码演示了如何用“resolveInstanceMethod:”来实现@dynamic实行;

id  autoDictionaryGetter(id self, SEL _cmd);
void    autoDictionarySetter(id self, SEL _cmd, id value);

+ (BOOL)resolveInstanceMethod:(SEL)selector  {
    NSStirng *selectorString = NSStringFromSelector(selector);
    if (/*selector is from a @dynamic property */) {
        if ([selectorString hasPerfix:@"set"]) {
            class_addMethod(self,
                            selector,
                            (IMP)autoDictionarySetter,
                            "v@:@");
        } else {
            class_addMethod(self,
                            selector,
                            (IMP)autoDictionaryGetter,
                            "@@:");
                }
        return YES;
        }
    }
return [super resolveInstanceMethod:selector];
}

首先将选择子转化为字符串,检测其是否表示设置方法。

备援接受者

- (id)forwardingTargetForSelector:(SEL)selector

完整的消息转发

首先创建NSInvocation 对象,把与尚未处理的那条消息有关的全部细节都封于其中。此对象包括 selector、target、及参数在触发NSInvocation 对象时,“消息派发系统”(message-dispatch system)将亲自出马,把消息指派给目标对象。

- (void)forwardInvocation:(NSInvocation *)invocation

消息转发全流程

以完整的例子演示动态方法解析