Runtime(三)消息转发

Message Forwarding

向一个未处理响应消息的对象发送对应的消息,会发生错误。但是,runtime会给这个对象第二次机会来处理这个message,避免发生错误。

Forwarding(消息转发)

向一个未处理响应消息的对象发送对应的消息,在通知错误前,runtime会向receiving object发送一个带有NSInvocation的参数的forwardInvocation:消息,这个NSInvocation参数包装了最初的消息(origin message)和参数。

你可以实现forwardInvocation:,给一个默认恢复(default response)或用其他方法阻止错误通知。正如其名字暗示forwardInvocation:将消息转发给其他对象。

为了验证转发(forwarding)的范围和意图,想象一下:首先,你创建了一个对象能够响应negotiate,并希望自己的响应(response)包含其它对象的响应。你在negotiate的实现中,把negotiate消息发送给其他对象。

更近一步:让自己对negotiate的响应由另一个类的实现去响应。一种方法是让你的类继承另一个类的方法,然后这是不可能的。因为不同的类可能在继承结构中的不同分支上。

尽管你不可以继承另一个类的方法,但是你仍然可以借它(borrow it):把message发送给其他对象。

- (id)negotiate {
  if ([someObject responseTo:@selector(negotiate)]) {
    return [someObject negotiate];
  }
  return self;
}

以上方法可能有一些笨重:你需要实现方法去覆盖所有你借的方法,对于整个消息集合(the full set of messages)你可能覆盖不了。再一个,在运行期,这些消息集合可能会改变的。

forwardInvocation:可以减少实现方法,并且是动态的。它是这样工作的:当一个object不可以response一个message的时候,runtime会发送forwardInvocation:message给object。每一个object继承NSObject的forwardInvocations:,可以重写(override)将messages发送给其他对象。NSObject的forwardInvocation:简单的调用doesNotRecognizeSelector:。

- (void)forwardInvocation:(NSInvocation *)anInvocation {
  if ([someOtherObject responseSelector:[anInvocation selector]])
    [anInvocation invokeWithTarget:someOtherObject];
  else
    [super forwardInvocation:anInvocation];
}

被转发的消息的返回值,会被返回最初的发送者(sender)

forwardInvocation:有这么几个角色:

  • 未识别消息(unrecognized messages)的分发中心(distribution center),打包messages给接受者(receiver)。
  • 或者是一个转移站(transfer station),把所有messages发送给相同的目的地。
  • 或者是把一个message转换为另一个message,或者简单的“吞咽”(swallow)一些messsages,所以没有response和error。
  • 或者把多个消息合并(consolidate)为一个响应。

forwardInvocation:能做什么取决于(up to)实现者(implementor)。

note:只有在不能调用名义上的(nominal)receiver的存在的方法,forwardInvocation:才起作用。如果你自己可以实现响应的方法,message永远不会到达forwardInvocation:

如果想了解更多forward,invocation,可以参考NSInvocation规格说明。

Forwarding and Multiple Inheritance(消息转发和多重继承)

Runtime(三)消息转发_第1张图片
Figure 5-1

消息转发机制与多重继承:

  • 消息转发模拟多重继承,起到了和多重继承一样的效果。一个对象通过转发机制response to message,就像“borrow”或“inherit”其他类定义的方法。
  • 本来object1要响应的message,通过borrow object2的方法来响应message,仿佛(seem)是自己响应一样。
  • object1和object2处在继承结构(inheritance hierarchy)的两个分支(two branches)上,消息响应让object1看起来像是继承自object2一样。
  • 消息转发机制虽然提供了多重继承的大部分功能,但是两者仍然有一些重要的不同。多重继承把不同的能力结合到一个单一对象中,趋向于更大,更多面的对象;另一方面,消息转发把问题分解成更小的对象(smaller objects),但是能够以一种对发送方(sender)透明的方式把这些对象(objects)联系在一起。

Surrogate Objects(代理对象)

消息转发不仅仅模拟多重继承,它也可以创建轻量级的对象来代表(reperent)和覆盖(cover)实质的(substantial)对象。代理人(surrogate)作为其他对象的代理,并把消息几种发给它。

假设一下:你有一个object要处理大量的数据,比如:处理图片,或读取文件在磁盘上的内容。建立这个对象是很耗时的,所以你更希望在需要它的时候,或系统不忙的时候创建它。但同时,你又需要它作为占位符,以便程序中的其他对象可以正常工作。

在这种情况下,你初始化一个轻量级的代理(lightweight surrogate),而不是一个完整的object。当时间到来,把messages转发给surrogate。

一个很好的例子

Forwarding and Inheritance(消息转发和继承)

尽管forwarding模拟Inheritance,但是NSObject从来不会混淆(confuse)它们。方法例如:responseSelector:和isKindOfClass:只会在Inheritance hierarchy查找,不会在消息转发链中(forwarding chain)查找。

a Warrior object is asked whether it responds to negotiate message,the answer is NO, even though it can receive negotiate messages without error and respond to them, in a sense, by forwarding them to a Diplomat. (See Figure 5-1.)

if ([aWarrior responseToSelector:@selector(negotiate)])
  ...

在很多情况下,NO是正确的答案。但是,它也可能不是。如果你想通过forwarding去创建(set up)一个代理对象(surrogate object)
,或扩展(extend)一个类的能力(capabilities),消息转发机制(forwarding mechanism)应该像hierarchy那样透明(transparent)。如果你想让你的objects看起来继承了其他objects(收到转发消息的objects)的行为,你需要重写responseToSelector:和isKindOfClass:,方法中含转发的规则(forwarding algorithm)。

- (BOOL)responseToSelector:(SEL)selector {
  if ([super responseToSelector:selector]) {
    return YES;
  } else {
    /* Here, test whether the selector message can
        be forwarded to another object and whether that
        object can respond to it. Return YES if it can.  
        */
  }
}

除了responseToSelector:和 isKindOfClass:,instancesRespondToSelector:也应该反映(mirror)到forwarding algorithm。如果使用了协议(protocols),conformsToProtocol:也应该加入到列表里。类似,an object转发收到的remote messages,methodSignatureForSelector:要返回这个方法的描述。

- (NSMethodSignature*)methodSignatureForSelector:(SEL)selector
{
    NSMethodSignature* signature = [super methodSignatureForSelector:selector];
    if (!signature) {
       signature = [surrogate methodSignatureForSelector:selector];
    }
    return signature;
}

你可以把forwarding algorithm放在私密代码里。

note:这是很先进的(advanced)技术(technique),但是当其他方法无法解决问题的时候,再使用它。forwarding mechanism并不打算代替inheritance。如果你确定使用该技术时,你一定要完全理解 the behavior of the class doing the forwarding and the class you’re forwarding to.

这一节提及到的方法,在NSObject规范(specification)提及。invokeWithTarget:在NSInvocation中提及。

你可能感兴趣的:(Runtime(三)消息转发)