主要是学习Objective-C的“多继承”——消息转发博客。
具体的代码是 : OC2Nature项目中 SendMSG 目录, 欢迎下载查看,谢谢!
Objective-C 不支持多继承,但是也能做到多继承的效果,具体方式有:消息转发、delegate与protocol、类别。
Objective-C 与 消息转发 是什么关系呢?还有一个专业名词叫消息机制。 在 Objective-C 中的 消息机制 发生在运行时,与编译时没有任何关系。
Objective-C是一门动态语言,一个函数是由一个selector(SEL),和一个implement(IML)组成的。Selector相当于门牌号,而Implement才是真正的住户(函数实现)。和现实生活一样,门牌可以随便发(@selector(XXX)),但是不一定都找得到住户,如果找不到系统会给程序几次机会来程序正常运行,实在没出路了才会抛出异常。
引用中提到的门牌可以随便发,是这个意思:
[receiver performSelector:@selector(message)];
在上面的message在代码中可以顺便写,在编译阶段不会报错,只会有警告而已,所以才说门牌可以随便发。其中 performSelector: 是系统发送消息的方法, message 就是所谓的门牌号。上面的语句可以理解成 拿着 message 这个门牌号到receiver对象中去寻找implement(IML)(具体实现),如果找到了就执行,如果没有找到就奔溃。
还有一种说法:每发送一个消息(message),知道程序运行的时候才会去寻找具体的实现(IML),在Objective-C中国年发送一条消息的经典表达式如下:
[receiver message];
最终在运行时转成了这样的格式:
objc_msgSend(receiver, @selector(message))
其中 objc_msgSend 可以有参数,如:
objc_msgSend(receiver, @selector(message), arg1,arg2)
是怎么通过门牌号去寻找具体的IML的呢?Objective-C的“多继承”——消息转发是使用下图解释的:
动态消息解析
快速消息转发
主要是重写如下系统方法:
- (id)forwardingTargetForSelector:(SEL)aSelector
按照我的理解应该是这样的:当前对象收到一个aSelector的时候, 先检查一下自己能否执行, 若是不能执行的话,就在这里交给别人执行吧。这个方法执行是有条件的,在动态方法没有找到的情况下才会被执行。
// Teacher 类需要实现将消息转发给 Doctor
- (id)forwardingTargetForSelector:(SEL)aSelector {
// 医生
Doctor *doctor = [[Doctor alloc]init];
if ([doctor respondsToSelector:aSelector]) {
return doctor;
}
// 返回 nil 说明快速消息转发失败
return nil;
}
标准消息转发
当快速消息转发返回 nil 的时候,会到标准消息转发。主要是重写以下两个系统方法:
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
- (void)forwardInvocation:(NSInvocation *)anInvocation
以下是参考截图
一个小 Demo
以上均是参考各位大神的博客的抄袭总结,由于这些方法通常也用不到,过一段时间来看的话感觉又模糊了,故特弄了一个 Demo,以便后期方便查看。具体的代码是 : OC2Nature项目中 SendMSG 目录, 欢迎下载查看,谢谢!
参考
0、Objective-C的“多继承”——消息转发
1、苹果
2、继承自NSObject的不常用又很有用的函数(2)
3、消息机制