OC runtime 方法调用转发机制

消息发送的正常处理流程

  - (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    Person *p = [Person new];
    // 调用 p 身上一个不存在的方法,就会进入消息转发
    SEL sel = NSSelectorFromString(@"caonima");
    [p performSelector:sel];
}
  1. 首先根据当前对象 reciver 对象的 isa 指针获取它的类型。
  2. 优先在 classcache 找到 message 方法,如果找不到,就去 methodList 方法列表里面去找。
  3. 如果在 class cache & methodList 都没有找到,会去 superClass 里面去找。
  4. 一旦找到了 message 方法,就执行该方法的 IMP
OC 方法搜索链条

消息的处理和转发流程(异常处理)

如果按照上述流程,仍然无法找到 message 的实现,消息就会进入转发阶段。(也就是在报 unrecognized selector sent to instance 0xxxxxx之前的阶段)

进入“异常处理”的阶段,也叫做消息转发

OC 消息转发流程

如果一个对象以及它的继承链条上都没有找到这个消息,那么就会进入到消息转发的阶段。

消息转发的第一个阶段:+ (BOOL)resolveInstanceMethod:(SEL)sel

当前 SEL 会转发给当前调用此方法的类。确认,自己是否真的无法完成此条消息的处理。
在这个方法中,返回一个 YES。告知系统,我可以处理这个消息。并动态的为此类添加这个方法。

//SEL 是结构体指针  IMP 是函数指针变量
void myMethodImp(id self,SEL _cmd) {
    NSLog(@"%@",@"我是 person 的方法实现");
}
+ (BOOL)resolveInstanceMethod:(SEL)sel {
    NSString *selName = NSStringFromSelector(sel);
    if ([selName isEqualToString:@"caonima"]) {
        class_addMethod([self class], sel, (IMP)myMethodImp, "v@:");
        return YES;// 告诉外界,我可以执行这个 sel,你在返回去执行一次吧。
    }

    return NO; // 告诉,我无法执行这个 sel。消息转发进入到下一个步骤。
}

运行结果:

2017-10-14 17:27:52.548 CodeFor消息转发[81717:21305980] 我是 person 的方法实现

或者直接返回一个 nil,告知系统,我无法处理这个消息,你接着去下一个阶段吧
消息转发就进入到第二步:- (id)forwardingTargetForSelector:(SEL)aSelector

消息转发第二步:- (id)forwardingTargetForSelector:(SEL)aSelector

如果消息转发第一步返回 nil,那么就会进入到此步。

/**
 告诉,系统,Fahter可以执行这个方法
 */
- (id)forwardingTargetForSelector:(SEL)aSelector {
    NSString *strSel = NSStringFromSelector(aSelector);
    if ([strSel isEqualToString:@"caonima"]) {
        return [Father new];
    }

    return nil;// 告诉系统,我也无法找到一个可以执行此方法的对象
}

运行结果:

2017-10-14 17:38:01.768 CodeFor消息转发[81937:21345385] 我是 father,我可以执行 caonima 这个消息

这一步的意义:作为接收消息的我来说,无法处理这个消息。但是我知道谁可以处理。

当然,如果当前我也不知道,谁可以处理这个消息,那么直接返回 nil
消息转发将会进入到第三步骤。

消息转发的第三步:- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector & - (void)forwardInvocation:(NSInvocation *)anInvocation

- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
动态的拼接一个Objc 方法的签名。如果返回 nil 或者一个错误的方法签名,将执行 doesNotRecognizeSelector

- (void)doesNotRecognizeSelector:(SEL)aSelector {
    NSLog(@"%@",@"没有返回一个正确的方法签名!");
    [super doesNotRecognizeSelector:aSelector];
}

2017-10-14 17:58:07.296 CodeFor消息转发[82176:21394061] 没有返回一个正确的方法签名!
2017-10-14 17:58:07.296 CodeFor消息转发[82176:21394061] -[Person caonima]: unrecognized selector sent to instance 0x618000009720

- (void)forwardInvocation:(NSInvocation *)anInvocation
动态的创建一个符合这个签名的方法,然后执行。

消息转发的第三步可以这么理解:

既然我自己无法处理这个消息,也不知道谁可以处理这个消息,那么我就自己创造一个可以处理这个消息的 NSInvocation.

一张更为清晰的消息转发机制流程图

关于 NSInvocation 如何自己创造一个对象方法来处理消息转发

苹果官方给的解释是:NSInvocation 作为对象呈现的 Objective-C 消息。

An Objective-C message rendered as an object.
NSInvocation objects are used to store and forward message between objects and between applications,
primarily by NSTimer objects and the distributed objects system.
An NSInvocation object contains all the elements of an Objective-C message: a target ,a selector, arguments , and the return value. Each of these elements can be set directly , and the return value is set automatically when the NSInvocation object is dispatched.

一个 NSInvocation 对象包含一个 Objective-C 消息的所有元素:target、selector、arguments、returnValue。可以直接设置每个元素,并在 NSInvocation 分派时,自动设置返回值。

// 一个完整方法OC方法调用
NSString *returnValue = [p runWithDestination:@"北极熊"];
一个完整的方法调用结构
  1. target
    2.sel
    3.arguments
    4.returnValue

使用消息转发的代码描述则是:

NSString *value = ((id(*)(id,SEL,id))objc_msgSend)(p,sel_registerName("runWithDestination:"),@"北极熊");
NSLog(@"%

OC方法和 C 消息发送对比

OC方法和 C 消息发送对比
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,569评论 4 363
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,499评论 1 294
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,271评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,087评论 0 209
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,474评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,670评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,911评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,636评论 0 202
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,397评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,607评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,093评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,418评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,074评论 3 237
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,092评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,865评论 0 196
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,726评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,627评论 2 270

推荐阅读更多精彩内容