美文网首页iOS面试iOS开发
OC的消息转发机制

OC的消息转发机制

作者: 莫道别离伤 | 来源:发表于2015-09-11 13:59 被阅读1906次

OC的消息转发机制众所周知,OC中的方法调用是利用消息转发实现的。

[obj foo] 等同于 objc_msgSend(obj,@selector(foo))

首先我们来了解一下类的底层构造如下:

struct objc_class {
 Class isa OBJC_ISA_AVAILABILITY; //isa指针   
#if !__OBJC2__ Class 
super_class OBJC2_UNAVAILABLE; // 父类 
const char *name OBJC2_UNAVAILABLE; // 类名 
long version OBJC2_UNAVAILABLE; // 类的版本信息,默认为0 
long info OBJC2_UNAVAILABLE; // 类信息
long instance_size OBJC2_UNAVAILABLE; // 类占据的内存大小 
struct objc_ivar_list *ivars OBJC2_UNAVAILABLE; // 成员变量链表 
struct objc_method_list **methodLists OBJC2_UNAVAILABLE; // 方法链表 
struct objc_cache *cache OBJC2_UNAVAILABLE; // 方法缓存列表 
struct objc_protocol_list *protocols OBJC2_UNAVAILABLE; // 协议链表
 #endif
 } OBJC2_UNAVAILABLE;

objc在向一个对象发送消息时,runtime库会根据对象的isa指针找到该对象实际所属的类,然后在该类中的方法列表以及其父类方法列表中寻找方法运行。如果在层层的寻找中,均未找到方法的实现,就是会抛出unrecognized selector sent to XXX的异常,导致程序崩溃。

在这之前OC的运行时提供了三次拯救程序的机会。

注:下述代码均放在 为可能出现方法非法调用的类书写的类目中。

  1. Method resolution

     pragma mark 消息转发第一步(实例) 
      //实例方法 
     + (BOOL)resolveInstanceMethod:(SEL)sel { 
     //可利用sel得到具体调用的方法,进而为方法列表添加不同的方法 
     class_addMethod([self class], sel, (IMP)test, "Test"); 
     return [super resolveInstanceMethod:sel];
      } 
     如此便达到了,当此类调用未定义的实例方法时,自动调用test函数,而避免了崩溃的情况。 
    

    类方法亦如此。

  2. Fast forwarding (方法转发)

      #pragma mark 消息转发第二步, 第一步失败后执行                     
     #pragma mark 其实只要返回对象不为self 和 nil 就会把消息转发给返回的对象 
     - (id)forwardingTargetForSelector:(SEL)aSelector { 
     NSString * str = NSStringFromSelector(aSelector); 
     NSString * obj = [NSString stringWithFormat:@"1234567"]; 
     NSLog(@"方法 %@ 即将转发给 Class %@",str,[obj class]); 
     return obj; 
     }
    
  3. Normal forwarding

首先会调用- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector 方法,倘若返回值为nil,则runtime会发出doesNotRecognizeSelector:消息,引发异常,程序崩溃。
如果返回了一个合理的函数签名,Runtime就会创建一个NSInvocation对象并发送-forwardInvocation:消息给目标对象。

    #pragma mark 消息转发第三步,前两步失败后到这里 
    - (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector { 
    /** *  参数列表 
    *  = char * 
    char BOOL = c
     : = SEL 
    ^type = type *
     @ = NSObject * 
    ^@ = NSError ** 
    # = NSObject 
    等等签名信息 
    然而我并不知道如何解析~~ 
    */ 
    //生成一个签名返回 可以和SEL毫无关系 
    return [NSMethodSignature signatureWithObjCTypes:"@@:"]; } 

    //返回签名后调用此方法,在参数中已经存储了SEL信息,简易的异常处理,可提示所有类的非法方法调用,不崩溃。 
    - (void)forwardInvocation:(NSInvocation *)anInvocation { 
    NSString * key = NSStringFromSelector([anInvocation selector]); 
    NSLog(@"Class %@ can't responer %@ methond",[self class],key);
     } 

  至此我不由想到,如果为NSObject书写一个类目,覆写上述方法,则就不会出现`unrecognized selector sent to XXX`的异常导致程序崩溃了。

事实如此。。但是并没有什么卵用~~

不过在请求数据时候,数据为NSNull时很容易引发上述异常,解决方案就是运用了消息转发机制的最后一道守护。

相关文章

  • 关于Runtime 消息发送机制的延伸

    说到OC 不得不说一下OC 的消息转发机制;何为OC 的消息转发机制;其实就是这样的; Objc 在向一个对象发送...

  • OC消息机制,消息转发机制

    Runtime简称运行时,其中最主要的是消息机制 概述 C 与 OC 的不同 1.C 语言,函数的调用在编译的时候...

  • Runtime知识点整理1

    OC消息机制?消息转发机制流程?什么是Runtime?什么场景下使用? ==============巴拉巴拉......

  • oc消息转发机制

    一、消息转发机制 在OC中,调用一个对象的方法,实际上是给对象发了一条消息,在编译Objective-C函数调用的...

  • OC消息转发机制

    当一个对象收到它没实现的消息的时候,通常会发生如下的情况。 调用+(BOOL)resolveInstanceMet...

  • OC 消息转发机制

    首先了解几个概念:class 的定义 method的定义 消息转发本质:在运行时将方法地址(imp)和一个名字(s...

  • OC 消息转发机制。

    当调用一个 NSObject 对象不存在的方法时,并不会马上抛出异常,而是会经过多层转发,层层调用对象的-reso...

  • OC 消息转发机制

    首先我们看一下objc_msgSend它具体是如何发送消息: 首先根据receiver对象的isa指针获取它对应的...

  • OC消息转发机制

    消息转发的两大阶段 先征询接收者,所属的类,看其是否能够动态添加方法,以处理这个『未知的选择子』,这叫做动态方法解...

  • OC消息转发机制

    暂时先看这篇文章消息转发

网友评论

    本文标题:OC的消息转发机制

    本文链接:https://www.haomeiwen.com/subject/esppcttx.html