(十二) [OC高效系列]消息的派发机制

1.什么是消息转发机制

  • 消息转发机制是在调用未知方法时出现的
  • 消息转发机制让程序员有机会去处理未知方法
  • 消息转发机制触发结束,如果该未知方法还是未处理则会触发crash
unrecognized selector send instance 

2.消息转发机制的流程

如图

消息转发机制

3.发现未知方法,在resolveInstanceMethod方法中使用runtime动态的添加一个实现

如代码,.m文件中并没有实现eat方法,而是在消息派发的过程中动态添加。

.h
@interface Person : NSObject

- (void)eat;

@end

.m
void eat(id self){
    NSLog(@"吃饭啊吃饭");
}

@implementation Person

+ (BOOL)resolveInstanceMethod:(SEL)sel{
    NSString *selectorStr = NSStringFromSelector(sel);
    
    if([selectorStr isEqualToString:@"eat"]){
        class_addMethod(self, sel, (IMP)eat, "v");
        return YES;
    }
    return NO;
}

@end

调用

 Person *p = [[Person alloc] init];
 [p eat]; //打印 吃饭啊吃饭

4.发现未知方法在forwardingTargetForSelector方法中,将消息派发给其他对象

//Boss
@interface Boss : NSObject
@property (nonatomic,strong) Programer *manong;

- (void)madaima;
- (void)jiaban;

@end

@implementation Boss

- (instancetype)init
{
    self = [super init];
    if (self) {
        _manong = [Programer new];
    }
    return self;
}

- (id)forwardingTargetForSelector:(SEL)aSelector{
    NSString *selStr = NSStringFromSelector(aSelector);
    if([selStr isEqualToString:@"madaima"]){
     
        NSLog(@"我是boss 我怎么可能亲自码代码,我的小小程序员赶快码去");
        
        return self.manong;
        
    }else if([selStr isEqualToString:@"jiaban"]){
        NSLog(@"我是boss,我怎么可能加班,程序员赶快加班干活");
        return self.manong;
    }
    return nil;
}

@end

//码农
@interface Programer : NSObject
- (void)madaima;
- (void)jiaban;

@end
@implementation Programer
- (void)madaima{
    NSLog(@"我是程序员,我在苦逼的码代码");
}
-(void)jiaban{
    NSLog(@"我是程序员,我在苦逼的加班");
}
@end

调用

Boss *b = [[Boss alloc] init];
[b jiaban];
[b madaima];

打印

2016-08-13 15:49:53.229 DashTestOC[17866:24254656] 我是boss,我怎么可能加班,程序员赶快加班干活
2016-08-13 15:49:53.230 DashTestOC[17866:24254656] 我是程序员,我在苦逼的加班
2016-08-13 15:49:53.231 DashTestOC[17866:24254656] 我是boss 我怎么可能亲自码代码,我的小小程序员赶快码去
2016-08-13 15:49:53.231 DashTestOC[17866:24254656] 我是程序员,我在苦逼的码代码

5.在实现forwardInvocation之前,必须实现methodSignatureForSelector方法

  • 这样做的目的是告诉派发机制该未知的方法签名,用以封装NSInvocation对象,否则会报错。
  • methodSignatureForSelector方法返回一个NSMethodSignature对象,此对象封装了该方法的返回类型和参数信息。

以此方法为例

id autoDictionary(id self,SEL _cmd){
    YXAutoDictionary *dic = (YXAutoDictionary *)self;
    NSString *key = NSStringFromSelector(_cmd);
    return [dic.dic objectForKey:key];
}

为此方法创建一个NSMethodSignature的代码为

[NSMethodSignature signatureWithObjCTypes:"@@:"]
  • 第一个@代表返回值是一个id类型的对象
  • 第二个@代表第一个参数是id类型的对象
  • 第三个:代表第二个参数是selector类型

其他objcType语义:

其他objcType语义

6.发现未知方法在forwardInvocation中,可以在这个方法中改变消息的参数,调用对象目标等

代码如下


//Boss
@interface Boss : NSObject
@property (nonatomic,strong) Programer *manong;

- (void)madaima:(int)lineNumber;

@end


@implementation Boss

- (instancetype)init
{
    self = [super init];
    if (self) {
        _manong = [Programer new];
    }
    return self;
}

- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{
    return [self.manong methodSignatureForSelector:aSelector];
}

- (void)forwardInvocation:(NSInvocation *)anInvocation{
    
    NSString *sel = NSStringFromSelector(anInvocation.selector);
    
    if([sel hasPrefix:@"madaima"]){
        int arg1;
        [anInvocation getArgument:&arg1 atIndex:2]; //注意这里要从2开始,因为0是self,1是SEL
        if(arg1<10000){
            NSLog(@"才写%d行代码,多写点,写50000行吧",arg1);
            int newArg = 50000;
            [anInvocation setArgument:&newArg atIndex:2];
            [anInvocation invokeWithTarget:self.manong];
        }
    }
    
}

@end

//Programer
@interface Programer : NSObject
- (void)madaima:(int)lineNumber;

@end

@implementation Programer

- (void)madaima:(int)lineNumber{
    NSLog(@"我是程序员,我在苦逼的要码%d行代码",lineNumber);
}
-(void)jiaban{
    NSLog(@"我是程序员,我在苦逼的加班");
}

@end

调用场景

    Boss *b = [[Boss alloc] init];
    [b madaima:10];

打印


2016-08-13 18:27:16.926 DashTestOC[18620:24313077] 才写10行代码,多写点,写50000行吧
2016-08-13 18:27:16.926 DashTestOC[18620:24313077] 我是程序员,我在苦逼的要码50000行代码

7.OC中的Runtime

  • OC中的Runtime可以动态的向一个类中添加方法,替换方法,添加属性,遍历方法属性等等。很多非常方便强大的框架,UI组件都是用Runtime实现的,而且可以大大减少代码的侵入性,使代码更大程度的解耦。
  • OC通过Runtime还可以实现伪继承,甚至是多重继承。
  • OC中的Runtime其实和Java、c#、js中的反射很相似。具体可以看我的另外一篇文章,对四种语言进行了对比。《c#中的反射》
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,826评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,968评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,234评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,562评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,611评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,482评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,271评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,166评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,608评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,814评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,926评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,644评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,249评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,866评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,991评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,063评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,871评论 2 354

推荐阅读更多精彩内容