C语言通俗理解消息的传递及转化机制

前言

于面试过程被你或会为咨询到消息转发机制。这篇稿子就是针对性信息的中转机制进行一个梳。主要包括什么是信、静态绑定/动态绑定、消息的传递及消息之转向。接下来开始进入正题。

信之讲

在其他语言里,我们得据此一个近乎去调用某个方法,在OC里面,这个法就是信息。某个类调用一个办法就是望此类似发送一长信息。举个例子:

People *zhangSan = [[People alloc] init];
People *lisi = [[People alloc] init];
[zhangSan beFriendWith:lisi];

我们发出个People的切近,zhangSan这个实例发送了平等条beFriendWith:的消息。你恐怕还看罢这种调用方式:

[zhangSan performSelector:@selector(beFriendWith:) withObject:lisi];

彼目的和地方的一模一样,都是望zhangSan发送了同样漫漫beFriendWith:的信息,传人的参数还是lisi。
此大概介绍一下SEL和IMP:

SEL:类成员方法的指针,但与C的函数指针还未一致,函数指针直接保存了点子的地点,但是SEL只是措施编号。
IMP:函数指针,保存了法地址。

我们叫@selector(beFriendWith:)为信息之选项择子或者选择器。(A
selector identifying the message to send)

静态绑定/动态绑定

所谓静态绑定,就是以编译期就可知决定运行时所调用的函数,例如:

void printHello() {
    printf("Hello,world!\n");
}
void printGoodBye() {
    printf("Goodbye,world!\n");
}

void doTheThing(int type) {
    if (type == 0) {
        printHello();
    }else {
        printGoodBye();
    }
}

所谓动态绑定,就是在运行期才会确定调用函数:

void printHello() {
    printf("Hello,world!\n");
}
void printGoodBye() {
    printf("Goodbye,world!\n");
}
void doTheThing(int type) {
    void (*fnc)(void);
    if (type == 0) {
        fnc = printHello;
    }else {
        fnc = printGoodBye;
    }
    fnc();
}

在OC中,对象发送信息,就见面下动态绑定机制来控制要调用的章程。其实底层都是C语言实现之函数,当目标吸收信继,究竟调用那个方式了控制给运行期,甚至你吗可以直接在运行时改变方法,这些特点还设OC成为同山头动态语言。

信息的传递

优先押一下一致长简单的音讯:

id returnValue = [someObject messageName:parameter];

其中:
someObject叫做接收者(receiver)。
messageName叫做选择器(selector)
选择器和参数合起来成为信息(message)
当编译器看到就漫漫信息,就会更换成为一条标准的C函数:objc_msgSend,此时会见成:

id returnValue = objc_msgSend(someObject,@selector(messageName:),parameter);

objc_msgSend可以在objc里面的message.h中看到:
C语言 1
据悉官方注释可以见到:

When it encounters a method call, the compiler generates a call to one
of the functions objc_msgSend, objc_msgSend_stret,
objc_msgSendSuper, or objc_msgSendSuper_stret. Messages sent to an
object’s superclass (using the super keyword) are sent using
objc_msgSendSuper; other messages are sent using objc_msgSend.
Methods that have data structures as return values are sent using
objc_msgSendSuper_stret and objc_msgSend_stret.

它们的打算是向一个实例类发送一个包含简单归回值的message。是一个参数个数不自然的函数。当遇到一个方法调用,编译器会转移一个objc_msgSend的调用,有:objc_msgSend_stret、objc_msgSendSuper或者是objc_msgSendSuper_stret。发送给父类的message会使用objc_msgSendSuper,其他的信会利用objc_msgSend。如果艺术的返回值是一个结构体(structures),那么尽管会见使objc_msgSendSuper_stret或者objc_msgSend_stret。
首先单参数是:指向接收该信息之类似的实例的指针
第二个参数是:要拍卖的信息之selector。
旁的便是只要传的参数。
如此消息派发系统就是在接收者所属类中查找器方法列表,如果找到与选择器名称符合的道就是逾反其促成代码,如果搜索不顶,就再度其父类找,等找到适当的办法以跳反至贯彻代码。这里过反至实现代码这无异操作下了尾递归优化。
使该消息无法让该类或者其父类解读,就会起进行信息转发。

略知一二消息转发机制(message forwarding)
动态方法分析

不用将信转发机制想象得够呛麻烦,其实看罢下面的您不怕会意识,没有那么麻烦。
我们有些上会遇见这样的crash:
C语言 2
我们且掌握crash的缘由是People没有gotoschool这个法,但是你调用了该法,所以会来NSInvalidArgumentException,reason:

-[People gotoschool]: unrecognized selector sent to instance 0x1d4201780'

连着下去为我们省从发送信息及者crash的经过。前面消息的传递没有成功找到实现,所以会走至信息转发其中,我先以People类里面实现了这般一个方法:

void gotoSchool(id self,SEL _cmd,id value) {
    printf("go to school");
}
//对象在收到无法解读的消息后,首先将调用所属类的该方法。
+ (BOOL)resolveInstanceMethod:(SEL)sel {
    NSString *selectorString = NSStringFromSelector(sel);
    if ([selectorString isEqualToString:@"gotoschool"]) {
        class_addMethod(self, sel, (IMP)gotoSchool, "@@:");
    }
    return [super resolveInstanceMethod:sel];
}

然后还运行程序,你晤面意识没有crash了,而且顺利打印出来”go to
school”。
斯是啊独情景呢?先瞧这方式:

+ (BOOL)resolveInstanceMethod:(SEL)sel OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);
+ (BOOL)resolveClassMethod:(SEL)sel OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);

此办法是objc里面NSObject.h里面的不二法门。从字面理解就是是处理实例方法(处理接近措施)。下面左边图是针对那个的牵线:
C语言 3
它们的打算就给一个实例方法(给定的选择器)动态提供一个贯彻。注释也供了一个demo告诉我们怎样动态增长实现。
也就是说当消息传递无法处理的时候,首先会看一下所属类,是否能够动态增长方法,以拍卖当下一无所知之抉择择子。这个历程叫“动态方法分析”(dynamic
method resolution)。
此地我于动态方法分析这里动态添加了落实,然后程序即使不见面崩溃啦。
若果是近似方式,就调用resolveClassMethod:方法进行操作,和地方的resolveInstanceMethod一样的处理方式。
这里尚用到了calss_addMethod,后面会单独写首博客对其牵线。感兴趣之可优先活动查看API。

备援接收者

当动态方法分析并未实现或者无法处理的下,就会尽

- (id)forwardingTargetForSelector:(SEL)aSelector OBJC_AVAILABLE(10.5, 2.0, 9.0, 1.0, 2.0);

其一艺术呢是objc里面NSObject.h里面的办法。我本着People进行了之类处理:

- (id)forwardingTargetForSelector:(SEL)aSelector {
    NSString *selectorString = NSStringFromSelector(aSelector);
    if ([selectorString isEqualToString:@"gotoschool"]) {
        return self.student;
    }
    return nil;

}

自己在People里面加加了一个Student类实例,然后实现了forwardingTargetForSelector:方法。然后运行,奇迹地觉察先后也从来不崩溃。该办法的意图是(上图也出介绍):
返回一个对准匪识别信息处理的对象。如果实现了该措施,并且该措施没有回到nil,那么这返回的目标就是会作新的收目标,这个未知的信将会受新目标处理。通过这个方案,我们得以据此做来模拟多复继承的某些特点,比如我回到多单近乎的成,那么就比如继承多独八九不离十一样进行处理。在对外调用者来说,好像就是是欠目标亲自处理的这些信息。

信转发

当动态方法分析及备援接收者都不曾开展拍卖的话,就见面实施:

- (void)forwardInvocation:(NSInvocation *)anInvocation OBJC_SWIFT_UNAVAILABLE("");

以此艺术吧是objc里面NSObject.h里面的点子,我本着People进行如下处理:

- (void)forwardInvocation:(NSInvocation *)anInvocation {
    NSLog(@"%@ can't handle by People",NSStringFromSelector([anInvocation selector]));
}
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
    NSMethodSignature *sign = [NSMethodSignature signatureWithObjCTypes:"@@:"];
    return sign;
}

重新运行程序,发现先后没有崩溃,只不过打印出来了“gotoschool can’t handle
by People”。
forwardInvocation:方法是用信息转发让另外对象。
C语言 4
由注释看:对一个你的对象不识别的音讯进行响应,你得还写methodSignatureForSelector:方法,该措施返回一个NSMethodSIgnature对象,该对象涵盖了吃定选择器所标识方法的叙述。主要含有返回值的消息及参数信息。
贯彻forwardInvocation:方法时,若发现调用的message不是由本类处理,则补充调用超类的同名方法。这样具有父类均有机遇处理此消息,直到NSObject。如果最终调用了NSObject的法门,那么该方法就见面调用“doesNotRecognizerSelector:”,抛来特别,标明选择器最终未可知得到处理。也尽管是面的crash:NSInvalidArgumentException。
时至今日,整个消息转发全流程结束。
齐一个王图:
C语言 5

总结

收信人在列一样步都发生会对未知消息进行拍卖,一句子话:越早处理越好。如果能以率先步做扫尾,就不开展其他操作,因为动态方法解析会将之方式缓存。如果动态方法分析不了,就放置第二步备援接收者,因为第三步还要创建完整的NSInvocation。
于整来平等整个:
Q:说一下你掌握的音讯转发机制?
A:
先期会调用objc_msgSend方法,首先以Class中的缓存查找IMP,没有缓存则初始化缓存。如果无找到,则向父类的Class查找。如果一直查找到根类仍旧无落实,则实行信息转发。
1、调用resolveInstanceMethod:方法。允许用户以此时为该Class动态增长实现。如果发落实了,则调用并回YES,重新开objc_msgSend流程。这次目标会应此选择器,一般是因她就调用了了class_addMethod。如果照尚未兑现,继续下的动作。
2、调用forwardingTargetForSelector:方法,尝试找到一个能响应该消息之目标。如果获到,则直接把消息转发让它,返回非nil对象。否则回nil,继续下的动作。注意这里并非回self,否则会形成死循环。
3、调用methodSignatureForSelector:方法,尝试获得一个方法签名。如果获得不顶,则直调用doesNotRecognizeSelector抛来十分。如果能够博取,则回非nil;传为一个NSInvocation并传于forwardInvocation:。
4、调用forwardInvocation:方法,将第三步获取到的计签名包装成Invocation传入,如何处理就在及时中了,并赶回非nil。
5、调用doesNotRecognizeSelector:,默认的实现是丢来深。如果第三步没能够取一个道签名,执行该步骤

外附相关乱代码(里面有动态方法解析demo)。
转载请注明来源:http://www.cnblogs.com/zhanggui/p/7731394.html