iOS - runtime-01 消息转发机制。

在学习 runtime 的时候,给自己做了一个笔记,是关于消息转发机制的。先新建一个 Person 类,在 .h 文件声明一个方法: sendMessage。

//  Person.h
//  01-RuntimeSendMessage
//
//  Created by Mac on 2019/10/30.
//  Copyright © 2019 Mac. All rights reserved.
//

#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@interface Person : NSObject
- (void)sendMessage:(NSString *)msg;
@end

NS_ASSUME_NONNULL_END

但我不在 .m 文件里实现它。

//
//  Person.m
//  01-RuntimeSendMessage
//
//  Created by Mac on 2019/10/30.
//  Copyright © 2019 Mac. All rights reserved.
//

#import "Person.h"
#import <objc/runtime.h>
#import "SpareWheel.h"

// OC 消息转发机制

@implementation Person


@end

这个时候我去实例化这个类,然后调用声明却未实现的方法。

//
//  ViewController.m
//  01-RuntimeSendMessage
//
//  Created by Mac on 2019/10/30.
//  Copyright © 2019 Mac. All rights reserved.
//

#import "ViewController.h"
#import "Person.h"

@interface ViewController ()

@end

@implementation ViewController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    self.view.backgroundColor = UIColor.whiteColor;
    
    Person *p = [[Person alloc] init];
    [p sendMessage:@"hello"];
}
@end

结果可想而知,程序崩了。


WeChatd3bf8d4598bdb790c6a64fcac873c6a3.png

那怎么解决这个问题呢?这个时候就需要使用 runtime 的消息转发来解决,声明了,不需要实现也能跑起来。
在使用 runtime 之前需要在 Person.m 文件里导入它的头文件。

//
//  Person.m
//  01-RuntimeSendMessage
//
//  Created by Mac on 2019/10/30.
//  Copyright © 2019 Mac. All rights reserved.
//

#import "Person.h"
#import <objc/runtime.h>

OC 消息转发机制在我学习的过程中,我个人理解可以分为 3 个阶段,在这里如果有什么好的建议请及时提,大家相互学习。
第一阶段:动态方法解析
第二阶段:查找备用接收者
第三阶段:完整的消息转发。第三阶段又分两个小结: 1. 方法签名。2. 消息转发

1.先从第一个阶段,动态方法解析开始实现,需要实现 resolveInstanceMethod 这个方法,字面意思,翻译过来就是 解析实例方法

+ (BOOL)resolveInstanceMethod:(SEL)sel {

    NSString *methodName = NSStringFromSelector(sel);
    if ([methodName isEqualToString:@"sendMessage:"]) {
        // 动态的添加方法实现, sel ---- IMP 的实现
        return class_addMethod(self, sel, (IMP)sendMessage, "v@:@");
    }

    return NO;
}

这个方法需要传入一个 SEL 类型的参数,返回值为 BOOL 类型。那这个 SEL 是什么呢?就是我们的方法编号
解释一下这段代码:
1.获取方法名:NSString *methodName = NSStringFromSelector(sel);
2.判断是否有这个方法名,如果有,动态的添加方法实现。如果没有,返回 NO。
3.动态的添加方法实现要传四个参数,(self, sel, IMP,"v@:@")。前面两个参数很好理解,第三个参数 IMP是什么?其实这里就是填你的方法名,当然,不是你头文件声明的方法哦,而是你在 .m 文件实现的方法。

void sendMessage(id self, SEL _cmd, NSString *msg) {
    NSLog(@"msg = %@", msg);
}

可以看到这个写法很明显不是 OC 的写法,是 C 的写法。
那第四个参数是什么意思呢?其实就是上面用 C 语言写的函数对应的 4 个参数。v -> 返回值类型(void), @ -> id 类型(一般是 self), ":" ->: 方法编号, @ -> id 类型(NSString)
到这里,第一阶段就完成了,可以跑一下项目,发现程序并没有崩,而且还打印了 msg。

WechatIMG20.jpeg

到这里,第一阶段:动态方法解析就完成了。
2. 第二阶段:查找备用接收者
在执行第二阶段之前,我们需要创建一个 SpareWheel的OC文件,然后在 .m 文件里实现一个 sendMessage的方法。

//
//  SpareWheel.m
//  01-RuntimeSendMessage
//
//  Created by Mac on 2019/10/30.
//  Copyright © 2019 Mac. All rights reserved.
//

#import "SpareWheel.h"

@implementation SpareWheel
- (void)sendMessage:(NSString *)msg {
    NSLog(@"message = %@", msg);
}
@end

然后在 Person.m 文件导入刚创建的 SpareWheel.h 文件

//
//  Person.m
//  01-RuntimeSendMessage
//
//  Created by Mac on 2019/10/30.
//  Copyright © 2019 Mac. All rights reserved.
//

#import "Person.h"
#import <objc/runtime.h>
#import "SpareWheel.h"

然后实现forwardingTargetForSelector这个方法。

- (id)forwardingTargetForSelector:(SEL)aSelector {
    NSString *methodName = NSStringFromSelector(aSelector);
    if ([methodName isEqualToString:@"sendMessage:"]) {
        // 返回备用接收者
        return [SpareWheel new];
    }
    return [super forwardingTargetForSelector:aSelector];
}

可以看到在里面的代码几乎和上面的一样。
1.获取方法名:NSString *methodName = NSStringFromSelector(sel);。
2.判断是否有这个方法名,如果有, 返回备用接收者。如果没有,返回 [super forwardingTargetForSelector:aSelector],就是它自己本身。
然后把第一阶段的代码注释调,来看一下运行结果。


WeChat6e550872c0224e2d2d52886198ab9bc5.png

可以看到,第二阶段的我们也搞定了。

3. 第三阶段:完整的消息转发
上面提到分两个步骤,现实第一个步骤,方法签名 methodSignatureForSelector

// 1. 方法签名
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
    NSString *methodName = NSStringFromSelector(aSelector);
    if ([methodName isEqualToString:@"sendMessage:"]) {
        
        return [NSMethodSignature signatureWithObjCTypes:"v@:@"];
    }
    return [super methodSignatureForSelector:aSelector];
}

可以看到,实现的思路和第一第二阶段一样,"v@:@",这个在上面也有解释,这里就不多说了。第一步完成后,开始第二步,消息转发forwardInvocation

// 2. 消息转发
- (void)forwardInvocation:(NSInvocation *)anInvocation {
    SEL sel = [anInvocation selector];
    SpareWheel *sp = [SpareWheel new];
    if ([sp respondsToSelector:sel]) {
        [anInvocation invokeWithTarget:sp];
    }else {
        [super forwardInvocation:anInvocation];
    }
}

假如在备用接收者也没有找到,它就封装这个方法(sendMessage)的一些信息,然后给它转发出去,所有的信息都放在了 NSInvocation 这个类里面。下面解释一下这段代码:
1.获得方法编号:SEL sel = [anInvocation selector];
2.实例化一个对象(备用接收者,假设它也去备用接收者查找),SpareWheel *sp = [SpareWheel new];
3.判断这个类里面有没有这个方法的实现 ,if ([sp respondsToSelector:sel]) {}else{}
4.如果有,就指定这个方法的接收者为这个类, [anInvocation invokeWithTarget:sp];
5.如果没有,就走自己的方法,[super forwardInvocation:anInvocation];
最后,把第二阶段的代码注释,看一下运行结果,可以看到,它一样会去找我们的备用接收者,将 message 打印出来了。

WechatIMG51.png

4.最后还有一个问题,那如果通过上面的三个方法都没有找到,怎么才能保证程序不会崩溃呢?
这个时候我们可以实现这个方法doesNotRecognizeSelector

- (void)doesNotRecognizeSelector:(SEL)aSelector {
    NSLog(@"未知错误");
}

然后把上面的代码注释掉,来看一下运行结果。


WechatIMG53.png

可以看到,程序运行起来并没有任何问题,并且,还打印了相关的东西。到这里,关于 runtime 的消息转发机制就暂时结束了。
最后,我自己写了一个思维导图,可以根据思维导图进行对比。


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