• 设为首页
  • 点击收藏
  • 手机版
    手机扫一扫访问
    迪恩网络手机版
  • 关注官方公众号
    微信扫一扫关注
    迪恩网络公众号

iOS理解Objective-C中消息转发机制附Demo

原作者: [db:作者] 来自: [db:来源] 收藏 邀请

点击上方“iOS开发”,选择“置顶公众号”

关键时刻,第一时间送达!


最近在重温Effective Objective-C 2.0,这篇文章属于重温的产物吧,本文会通过demo来讲解OC中的消息转发机制



Demo:(https://github.com/maligh/ML-Objective-C-Demo/tree/master/MessageForwarding)

话不多说,iOS开发过程中我们经常会碰到这样的报错:unrecognized selector sent to instance **,原因是我们调用了一个不存在的方法。用OC消息机制来说就是:消息的接收者不过到对应的selector,这样就启动了消息转发机制,我们可以通过代码在消息转发的过程中告诉对象应该如何处理未知的消息,默认实现是抛出下面的异常


unrecognized selector.png


下面我们通过实例来看一下在抛出异常之前也就是消息转发过程中都经过了哪些步骤:


第一步:对象在收到无法解读的消息后,首先会调用+(BOOL)resolveInstanceMethod:(SEL)sel或者+ (BOOL)resolveClassMethod:(SEL)sel, 询问是否有动态添加方法来进行处理,处理实例如下


//People.m

void speak(id self, SEL _cmd){

    NSLog(@"Now I can speak.");

}

+ (BOOL)resolveInstanceMethod:(SEL)sel {

    

    NSLog(@"resolveInstanceMethod:  %@", NSStringFromSelector(sel));

    if (sel == @selector(speak)) {

        class_addMethod([self class], sel, (IMP)speak, "[email protected]:");

        return YES;

    }

    return [super resolveInstanceMethod:sel];

}


当People 收到了未知 speak选择子的消息的时候,如果是实例方法会首选调用上文的resolveInstanceMethod:方法,方法内通过判断选择子然后通过class_addMethod方法动态添加了一个speak的实现方法来解决掉这条未知的消息,此时消息转发过程提前结束。

但是当People 收到fly 这条未知消息的时候,第一步返回的是No,也就是没有动态新增实现方法的时候就会调用第二步


第二步:既然第一步已经问过了,没有新增方法,那就问问有没有别人能够帮忙处理一下啊,调用的是- (id)forwardingTargetForSelector:(SEL)aSelector这个方法

上文我们说到People接收到了一条选择子为fly的未知消息,我们可以看到控制台已经打印了resolveInstanceMethod: fly,代表第一步已经问过了,那么第二步问一下是否有别的类能帮忙处理吗?代码如下:


- (id)forwardingTargetForSelector:(SEL)aSelector {

    NSLog(@"forwardingTargetForSelector:  %@", NSStringFromSelector(aSelector));

    Bird *bird = [[Bird alloc] init];

    if ([bird respondsToSelector: aSelector]) {

        return bird;

    }

    return [super forwardingTargetForSelector: aSelector];

}

// Bird.m

- (void)fly {

    NSLog(@"I am a bird, I can fly.");

}


通过- (id)forwardingTargetForSelector:(SEL)aSelector的处理,bird能够处理这条消息,所以这条消息被bird成功处理,消息转发流程提前结束。控制台打印


forwardingTargetForSelector:  fly

I am a bird, I can fly.


但是如果- (id)forwardingTargetForSelector:(SEL)aSelector也找不到能够帮忙处理这条未知消息,那就会走到最后一步,这步也是代价最大的一步


第三步:调用- (void)forwardInvocation:(NSInvocation *)anInvocation,在调用forwardInvocation:之前会调用- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector方法来获取这个选择子的方法签名,然后在-(void)forwardInvocation:(NSInvocation *)anInvocation方法中你就可以通过anInvocation拿到相应信息做处理,实例代码如下


当People 收到一条 选择子为code 的消息的时候,前两步发现都没办法处理掉,走到第三步:


- (void)forwardInvocation:(NSInvocation *)anInvocation {

    NSLog(@"forwardInvocation: %@", NSStringFromSelector([anInvocation selector]));

    if ([anInvocation selector] == @selector(code)) {

        Monkey *monkey = [[Monkey alloc] init];

        [anInvocation invokeWithTarget:monkey];

    }   

}


- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {

    NSLog(@"method signature for selector: %@", NSStringFromSelector(aSelector));

    if (aSelector == @selector(code)) {

        return [NSMethodSignature signatureWithObjCTypes:"[email protected]:@"];

    }

    return [super methodSignatureForSelector:aSelector];

}


这时控制台会打印


resolveInstanceMethod:  code

forwardingTargetForSelector:  code

method signature for selector: code

forwardInvocation: code

I am a coder.


此时这个code消息已经被monkey实例处理掉

此时消息转发流程完整的结束了,完整的消息转发流程如下:



那么最后消息未能处理的时候,还会调用到

- (void)doesNotRecognizeSelector:(SEL)aSelector这个方法,我们也可以在这个方法中做些文章,避免掉crash,但是只建议在线上环境的时候做处理,实际开发过程中还要把异常抛出来


EOF:OC中消息转发流程大概就是这样了,Demo(https://github.com/maligh/ML-Objective-C-Demo/tree/master/MessageForwarding),由于个人能力有限,文中难免有些错误,希望大家不吝赐教~

另外有一个问题想问大家,+ (BOOL)resolveClassMethod:(SEL)sel 在这个方法中怎么动态添加类方法? 比如我发送了一条未知的 [People missMethod]消息,怎么添加 +(void)missMethod 的实现呢?



  • 作者:SuperMario_Nil

  • 链接:https://www.jianshu.com/p/f9bd98ad5b05

  • iOS开发整理发布,转载请联系作者授权

【点击成为Java大神】


鲜花

握手

雷人

路过

鸡蛋
该文章已有0人参与评论

请发表评论

全部评论

专题导读
上一篇:
十分钟让你明白Objective-C的语法(和Java、C++的对比)发布时间:2022-07-12
下一篇:
Objective-C语法之代码块(block)的使用发布时间:2022-07-12
热门推荐
热门话题
阅读排行榜

扫描微信二维码

查看手机版网站

随时了解更新最新资讯

139-2527-9053

在线客服(服务时间 9:00~18:00)

在线QQ客服
地址:深圳市南山区西丽大学城创智工业园
电邮:jeky_zhao#qq.com
移动电话:139-2527-9053

Powered by 互联科技 X3.4© 2001-2213 极客世界.|Sitemap