拜读yulingtianxia大神关于Runtime的博文后,针对消息发送这一块,做点小记录。
参考链接:Objective-C Runtime
objective-c中,[target selector] 在运行时会被Runtime系统转化为objc_msgSend函数。而该函数内其实做了一系列复杂的操作。
一.检查target是否为nil,如果是nil则忽略。
(objective-c中,任意一个nil对象发送消息都不会崩溃)
二.查找方法的实现
1.先从缓存中查找;【objective-c中的对象在运行时会被Runtime系统转化为结构体。在结构体中,有一个isa指针,它最终指向的就是对象所属的类。isa实际是一个指向objc_class结构体的指针,该结构体下关联了它的超类指针,类名,成员变量,方法,缓存,还有附属的协议。这里我们要说的就是缓存。当某个方法被调用之后,Runtime就会把该方法保存到cache中。】
2.从方法分发表中查找,找不到就顺着父类的分发表继续查找,直到NSObject;
3.分发表找不到则进入动态方法解析
三、动态方法解析
Runtime系统会调用resolveInstanceMethod: 或 resolveClassMethod:方法来给我们一次为类动态添加方法的机会。当这两个方法返回NO时,则进入消息转发。
+(BOOL)resolveInstanceMethod:(SEL)aSEL
{
if(aSEL==@selector(test)){
// // 该方法第一个参数为添加方法的类
// // 第二个参数为方法的声明
// // 第三个参数为方法的实现
// // 第四个参数为方法的类型即TypeEncodings。从左到右依次为返回值类型、参数类型。可参考官方文档https://developer.apple.com/library/mac/documentation/Cocoa/Conceptual/ObjCRuntimeGuide/Articles/ocrtTypeEncodings.html
// class_addMethod([self class], aSEL, (IMP)test, "v@:");
// return YES;
returnNO;
}
return[superresolveInstanceMethod:aSEL];
}
上述代码中,if语句内如果返回YES,则整个消息发送的过程到此为止。真正执行的就是test内部的实现。
四、消息转发
1.消息转发之前,可以通过-(id)forwardingTargetForSelector:(SEL)aSelector 实现重定向 如果该方法返回nil或者self,则进入消息转发机制
-(id)forwardingTargetForSelector:(SEL)aSelector
{
if(aSelector==@selector(test)){
// // 这里我在另一个类中声明了一个test的方法用以测试
// return [Test02 new];
returnself;
}
return[superforwardingTargetForSelector:aSelector];
}
上述代码中,如果返回的是注释部分的代码,则消息发送过程到此为止,消息将被转发至[UIApplication sharedApplication].delegate对象。(从头开始消息发送过程)。
2.通过重写 forwardInvocation: 来实现转发逻辑。实际上在forwardInvocation: 消息发送之前,Runtime系统会向对象发送 methodSignatureForSelector: 来获取返回的方法签名用于生成NSInvocation对象。 所以在重写forwardInvocation: 方法的同时也需要重写methodSignatureForSelector: 方法,并且返回正确的方法签名。否则会抛出无法识别方法的异常。
// 消息转发
-(void)forwardInvocation:(NSInvocation*)anInvocation
{
if(anInvocation.selector==@selector(test)){
[anInvocation invokeWithTarget:[Test02new]];
}else{
[superforwardInvocation:anInvocation];
}
}
// 返回方法签名
-(NSMethodSignature*)methodSignatureForSelector:(SEL)aSelector
{
// return [NSMethodSignature signatureWithObjCTypes:"v@:"];
// 签名对象记录着方法的参数和返回值的类型信息。
NSMethodSignature*signature=[supermethodSignatureForSelector:aSelector];
if(!signature){
// 本例中,因为要将消息转发给Test02对象,因此我们直接实例化一个Test02对象,并获取其同名方法的签名,并返回
signature=[[Test02new]methodSignatureForSelector:aSelector];
}
returnsignature;
// return nil;
}
注:动态添加方法,消息转发时,最好确保参数及返回值等信息一致。
如上述代码中,若获取方法签名的方法返回nil则抛出无法识别方法的异常;
若返回的签名信息不符,签名信息多余仍然可以执行,只是有可能发生“灵异事件”。若签名信息较少,则会抛出-[NSInvocation getArgument:atIndex:]: index (1) out of bounds [-1, 0]的错误。因为在调用方法时,从签名信息中获取参数和返回值信息就会出现越界情况了。
附下载链接—测试的小demo
网友评论