Objective-C消息转发

OC中调用方法就是向对象发送消息
下面的代码:

 [self performSelector:@selector(method) withObject:nil];

如果说method方法不存在,就会出现:unrecognized selector sent to instance
从发送消息到崩溃中间经历了什么?
首先分两种情况
1,如果发送的是一个实例方法,会经过三个步骤
第一步:

+ (BOOL)resolveInstanceMethod:(SEL)sel

这个方法是实例方法未实现时自动执行,这个方法里面是让你在当前类里面对未实行的方法做弥补

void dynamicMethodIMP(id sel,SEL _cmdd){
    NSLog(@"resolveInstanceMethod调用成功");
}

+ (BOOL)resolveInstanceMethod:(SEL)sel{
    if (sel == @selector(method)) {
//        class_addMethod([self class],sel,(IMP));
        class_addMethod([self class],sel,(IMP)dynamicMethodIMP, "v@:");
        return YES;
    }
    return [super resolveInstanceMethod:sel];
}

第二步:
如果第一步resolveInstanceMethod你没做处理,便会来到这一步,如果你处理了第一步,第二步是不会执行的

- (id)forwardingTargetForSelector:(SEL)aSelector{
    if (aSelector == @selector(method)) {
        return [TestClass new];
    }
    return nil;
}
@implementation TestClass
- (void)method{
    NSLog(@"method");
}
@end

这一步是在其他类里面找对应的方法,所以TestClass里面必须要实现method方法,不然同样会崩溃
第三步:
经历过第一步,在本类中找弥补方法,第二步,在其他类中找弥补方法后,如果都没有找到,便会来到第三步

- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
{
    NSMethodSignature *methodSignature = [super methodSignatureForSelector:aSelector];
    if (!methodSignature) {
        methodSignature = [NSMethodSignature signatureWithObjCTypes:"v@:*"];
    }
    return methodSignature;
}
- (void)forwardInvocation:(NSInvocation *)anInvocation{
    testClass *messageForwarding = [testClass new];
    if ([messageForwarding respondsToSelector:anInvocation.selector]) {
        [anInvocation invokeWithTarget:messageForwarding];
    }
}

这一步实现方法的重新签名
2.如果是调用了一个类方法,该类方法没有实现该怎么办
便会调用resolveClassMethod

void dynamicMethodIMP(id sel,SEL _cmdd){
    NSLog(@"resolveInstanceMethod调用成功");
}
@implementation testObject
+ (BOOL)resolveClassMethod:(SEL)sel{
  if (sel == @selector(hehe1)){
       class_addMethod(objc_getMetaClass("testObject"),sel,(IMP)dynamicMethodIMP, "v@:");
        return YES;
    }
    return [super resolveInstanceMethod:sel];
}
@end

看里面class_addMethod的第一个参数,是objc_getMetaClass("testObject"),该方法是获取当前类的元类,如果用[self class]是会崩溃的,因为dynamicMethodIMP是一个c方法,它是存在于元类里面的MessageList里面的,当前类找不到该方法,元类的知识可以看我另一篇文章:iOS类和元类关系
如果resolveClassMethod方法没有实现,就会直接崩溃,也就是说它没有实例方法里面的第二步和第三步。
resolveInstanceMethod和methodSignatureForSelector、forwardInvocation只是在实例方法缺失才会执行,类方法缺失不会执行,这点要注意

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 首先感谢这几个篇文章对我的帮助:http://blog.csdn.net/mangosnow/article/de...
    J_雨阅读 15,953评论 50 165
  • 通过一个小例子来简单、通俗的理解一下什么是消息转发以及如何消息转发首先,你需要知道这两个概念:OC中调用方法就是向...
    TobyStark阅读 3,290评论 0 0
  • 我们要通过一个小例子来简单、通俗的理解一下什么是消息转发以及如何消息转发,希望看完这篇文章时大家会彻底的明白OC的...
    Mr_Baymax阅读 3,114评论 1 2
  • 消息转发机制的出现 回想一下,在我们日常的程序开发中,是不是经常会遇到下面这种情况: 我们自定...
    是土豆啊丶阅读 4,084评论 0 4
  • 《问爱是什么?》 文/梦川 有人问爱是什么 我回答说:爱是 多少人走到最后 缘尽了,散了 多少人还没开始 缘也尽了...
    梦一直在路上阅读 2,299评论 0 1

友情链接更多精彩内容