iOS进阶之Runtime运行时(方法执行过程、拦截调用)

目录

前言

OC中发送消息:使用中括号把接收者和消息括起来。直到运行时才会把消息与方法实现绑定。
[person run]; 会被编译器转为 objc_msgSend(person,@selector(run));

objc_msgSend

/*
objc_msgSend(person, @selector(run))       
消息发送步骤:
  1. 检测 SEL 是否应该被忽略
  2. 检测 target 是否是nil ,为nil则忽略该消息。
      向nil发消息(不会崩)
        返回值是对象,返 nil
        返回值为指针类型,返回 0。
        返回值为结构体,返回 0(结构体中各个字段的值将都是0)。
        返回值以上都不是,返回 值未定义
  3. 查找IMP
    具体参见下面第1小节
*/

id objc_msgSend(id self,SEL op, ...);     
/*
根据实例对象和SEL 寻找 IMP
    // 第一个参数: id
    typedef struct objc_object *id;
    // 第二个参数: SEL
    typedef struct objc_selector *SEL;   
    // 第三个参数: 方法参数列表
*/

SEL

方法名的唯一标识
    一个方法名对应一个SEL(不同类相同方法的SEL相同)。
    OC中不支持函数重载就是因为一个类的方法列表中不能存在两个相同的 SEL 。
    SEL x1=@selector(method:);       
    SEL x2=NSSelectorFromString(@"method:");
    SEL x3=sel_registerName("method:");

typedef struct objc_selector *SEL;
struct objc_selector {
    char *name;                       OBJC2_UNAVAILABLE;
    char *types;                      OBJC2_UNAVAILABLE;
};

IMP

方法具体实现的地址  

    typedef id (*IMP)(id, SEL, ...); (类实例或元类,SEL,参数)  函数地址
/*
instance -> class -> method -> SEL -> IMP -> 实现函数
实例对象中存放 isa 指针以及实例变量,有 isa 指针可以找到实例对象所属的类对象 (类也是对象,面向对象中一切都是对象),类中存放着实例方法列表,在这个方法列表中 SEL 作为 key,IMP 作为 value。 在编译时期,根据方法名字会生成一个唯一的 Int 标识,这个标识就是 SEL。IMP 其实就是函数指针 指向了最终的函数实现。整个 Runtime 的核心就是 objc_msgSend 函数,通过给类发送 SEL 以传递消息,找到匹配的 IMP 再获取最终的实现。

类中的 super_class 指针可以追溯整个继承链。向一个对象发送消息时,Runtime 会根据实例对象的 isa 指针找到其所属的类,并自底向上直至根类(NSObject)中 去寻找 SEL 所对应的方法实现,找到后就运行。

metaClass是元类,也有 isa 指针、super_class 指针。其中保存了类方法列表。
*/
对象的内存布局

1. 方法执行过程

1: 首先,在对象的缓存方法列表中寻找被调用的方法,如果找到则转向相应方法实现并执行。
    objc_cache
2: 如果没找到,在对象中的方法列表中寻找被调用的方法,如果找到则存添加到缓存列表并转向相应方法实现并执行。 
    objc_method_list
3: 如果没找到,去父类中继续执行1,2。
4: 以此类推,如果一直到根类还没找到则转向【拦截调用】。
5: 如果没有重写拦截调用的方法,程序报错。

2. 拦截调用(unrecognized selector崩溃前的一道关卡)

三次补救

当找不到相应方法,在未识别方法崩溃之前会转向拦截调用(有三次补救机会)。

注意:
  1. 有些个第三方内部也会实现下面几个方法,如果不小心被自己写的方法覆盖会出错

》》》》实例方法

当调用不存在的实例方法时首先会调用方法1

方法1 (默认返回false,然后去调用方法2(方法2未实现则崩溃))
    // 可添加方法后返回true
    + (BOOL)resolveInstanceMethod:(SEL)aSEL{
        if (aSEL == @selector(goToSchool:)) {
            class_addMethod([self class], aSEL, class_getMethodImplementation([self class], @selector(myInstanceMethod:)), "v@:");
            return YES;
        }
        return [super resolveInstanceMethod:aSEL];
    }
方法2 
    // 转发给拥有该方法的实例
    // 返回nil 则调用3(3未实现则崩)
    - (id)forwardingTargetForSelector:(SEL)aSelector{
        if(aSelector == @selector(mysteriousMethod:)){
            return [Person new];
        }
        return [super forwardingTargetForSelector:aSelector];
    }
方法3 
    // 作相应处理后,调用invokeWithTarget:将invocation传给拥有该方法的实例
    - (void)forwardInvocation:(NSInvocation *)anInvocation{
        anInvocation.target=nil;
        [anInvocation invocation];
        /*
        // invocation封装了原始的消息和消息的参数
        Person *person=[Person new];
        if ([person respondsToSelector:[anInvocation selector]]){
            [anInvocation invokeWithTarget: person];
        }else{
            [super forwardInvocation:anInvocation];
        }
        */
    }
    /* 
    调用forwardInvocation:之前,会先调用methodSignatureForSelector:,并取到返回的方法签名用于生成NSInvocation对象。
    在重写forwardInvocation:的同时必须重写methodSignatureForSelector:获取函数的参数和返回值,系统会自动创建NSInvocation并调用forwardInvocation:,否则会抛异常。
    返回nil会抛异常。
    */
    - (NSMethodSignature*)methodSignatureForSelector:(SEL)selector{
        NSMethodSignature* signature = [super methodSignatureForSelector:selector];
        if (!signature) {
            signature = [NSMethodSignature signatureWithObjCTypes:"v@:*"];
        }
        return signature;
    }

》》》》类方法

1 
    // 当调用不存在的类方法时调用(默认:false,调用2(2未实现则崩))
    // 可作相应处理后返回true
    + (BOOL)resolveClassMethod:(SEL)sel {
        if (sel == @selector(learnClass:)) {
            class_addMethod(object_getClass(self), sel, class_getMethodImplementation(object_getClass(self), @selector(myClassMethod:)), "v@:");
            return YES;
        }
        return [class_getSuperclass(self) resolveClassMethod:sel];
    }
2

例1

请求网络获取数据的时候会经常需要:responseObsect[@"data"][@"persons"]
万一后端数据返回的data数据类型不是字典还是数组,会导致未识别方法崩溃。

解决:在NSArray分类中添加
+(BOOL)resolveInstanceMethod:(SEL)aSEL{
    if (aSEL==@selector(objectForKeyedSubscript:)) {
        class_addMethod([self class], aSEL, class_getMethodImplementation([self class], @selector(myInstanceMethod:)), "v@:");
        return YES;
    }
    return [super resolveInstanceMethod:aSEL];
}
-(void)myInstanceMethod:(id)param{
    NSLog(@"objectForKeyedSubscript未识别方法崩溃捕捉");
}

例2

万一后端数据返回的data数据类型是null,会导致未识别方法崩溃。

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

推荐阅读更多精彩内容