[iOS面试]第3章 RunTime相关面试问题

本文主讲RunTime相关面试问题,包括数据结构、类对象与元类对象、消息传递、方法缓存、消息转发、Method-Swizzling、动态添加方法、动态方法解析。

一、类对象与元类对象

1) objc_object

objc_object结构

实际使用所有对象都是id类型, id对象代表就是objc_object结构体.
id = objc_object 分为以下几部分:

  • isa_t
  • 关于isa操作相关(如:获取isa所指向的类对象 或者 通过类对象isa获取它元类对象一些便利方法)
  • 弱引用相关 (如:标记一个对象是否标记过弱引用指针)
  • 关联对象相关(如: 这个对象设置关联属性)
  • 内存管理相关 (如:MRC retain release , ARC @autoreleasepool)

2) objc_class

objc_class结构

Class = objc_class
objc_class 继承自 objc_object, 所以Class也是一个对象

  • Class superClass (指向父类对象)
  • cache_t cache (方法缓存结构, 进行消息传递会使用这个数据结构)
  • class_data_bits_t bits (类定义的变量 属性和方法都在这个结构中)

3) isa指针

isa_t结构

共用体 isa_t (问题:isa指针是什么含义?)

  • 在32位或64位架构下,都是32或者64个0或者1的二进制数字 ,isa指针分为指针形isa和非指针形isa
  • 指针型isa的代表Class的地址
  • 非指针型isa的值的部分代表Class的地址

4) isa指针的指向

isa指向
  • 关于对象,其指向类对象
  • 关于类对象,其指向元类对象
  • 元类对象的isa指针都指向根元类对象,而根元类对象对象的isa指针指向根类对象。

方法调用时,调用实例方法实际上通过isa指针到类对象中进行方法查找.
如果调用类方法, 通过类对象isa这种到元类对象中进行方法查找.

5) cache_t

cache_t 特点:

  • 用于快速查找方法执行函数 (提高消息传递速度)
  • 是可增量扩展哈希表结构 (提高查找效率)
  • 局部性原理的最佳应用
cache_t数据结构

cache_t 理解为一个数组实现的, 里边存储 bucket_t结构体, bucket_t有两个成员变量. key对应OC中 @selector , IMP理解为无类型函数指针. 调用方法时使用SEL, 通过方法选择器名称来寻找具体实现IMP.

6)class_data_bits_t

  • class_data_bits_t主要是对class_rw_t的封装
  • class_rw_t 代表类相关的读写信息, 对class_ro_t的封装
  • class_ro_t代表类相关的只读信息

7) class_rw_t

class_rw_t数据结构

为一个类添加分类中的协议 属性 方法都在protocols properties methods 这三个结构中.这三个数据结构是一个二维数组(list_array_tt)

8) class_ro_t

class_ro_t数据结构

class_ro_t 中一维数组 ivars protocols properties methodList 存储的原始类定义添加的成员变量 协议 属性和方法列表

二、runtime整体数据结构

1) method_t

method_t结构

method_t结构体封装了函数四要素,其中名称通过SEL方法选择器表示,返回值和参数则由“Type Encodings”类型的字符串表示,函数体则指代了IMP函数指针。

types结构

更多关于Type Encodings

Type Encodings

2)runtime整体数据结构

runtime整体数据结构

三 实例对象、类对象、元类对象

  • 类对象存储实例方法列表等信息 的数据结构
  • 元类对象存储类方法列表等信息 的数据结构.

关于类对象的isa指针指向可以用下图表示:


实例对象 类对象 元类对象关系.png
  • Root class 是根类,分类父类指向nil, 实际指 NSObject这个类
  • 左侧部分指实例对象, 也就是objc_object这个数据结构,实例isa指向实例对象的类对象
  • 右侧部分指元类对象, 任何元类对象isa指针指向根元类对象,根元类对象自身isa指针指向根元类对象.根元类对象superclass指针指向根类对象
  • 当调用类方法从元类对象方法列表中逐级父类往上查找 , 查找到根元类对象(Root class meta)找不到时, 就会去根类(Root class class)对象中查找同名的实例方法实现.

问题:类对象和元类对象有什么区别和联系?
答:

  • 实例对象可以通过isa指针找到它的类对象
  • 类对象存储实例方法列表等信息,类对象可以通过它的isa指针找到它的元类对象,从而可以访问类方法列表等信息.
  • 类对象和元类对象都是objc_class数据结构,objc_class数据结构由于继承objc_object,所以类对象和元类对象才有isa指针.进而实例对象可以通过isa指针找到对应类对象,访问实例方法列表等信息, 类对象通过isa指针找到元类对象,访问类方法列表等信息.

问题:如果调用类方法没有对应的实现, 当时有同名的实例方法实现, 这个时候会不会发生崩溃?会不会产生实际调用?
答: 由于根元类对象的superclass指针指向了根类对象, 当查找到根元类对象(Root class meta)类方法找不到时, 就会去根类(Root class class)对象中查找同名的实例方法实现,如果找到调用.

四、消息传递机制

1) 消息传递流程

可以用下图展示消息传递的流程:

消息传递流程

注意:在消息缓存中查找是通过哈希表来快速定位函数指针,而在当前类方法列表中查找时,对于已经排序好的列表使用二分查找,而对于没有排序的列表采用一般遍历查找法。

2) 缓存查找

例 :给定值是SEL, 目标值是对应的bucket_t中的IMP.


缓存查找

问题:缓存查找具体的是怎样的流程和步骤?
答:缓存查找实际上就是从 cache_t中 把对应bucket_t找出来.
根据给定的方法选择器,通过一个函数来映射出bucket_t在数组中映射的位置, 实际上就是哈希查找. 哈希查找通过给定的值, 经过哈希函数算法算出的值, 实际为给定值在数组中的索引位置.

3)当前类中查找

  • 对于已排序好的列表, 采用二分查找算法查找方法对应执行函数.
  • 对于没有排序的列表, 采用一般遍历查找方法对应执行函数.

4)父类逐级查找

父类逐渐查找流程

问题: 消息传递机制?
答:
1)缓存是否命中, 当前类方法列表是否命中, 逐级父类方法列表是否命中
2)根据三个方面分别讲述具体情况

五、消息转发流程

消息转发流程

resolvelnstanceMethod方法中为对象动态添加方法,已达到处理消息未被实现的问题。

objc_msgSend方法调用找不到响应的函数名称时就会进行消息转发,主要分为3步:
1、动态方法解析
调用方法+(BOOL)resolveInstanceMethod:(SEL)sel(实例方法动态解析)和+ (BOOL)resolveClassMethod:(SEL)sel(类方法动态解析)。

2、备援接收者
调用方法 - (id)forwardingTargetForSelector:(SEL)aSelector

3、完全转发
调用方法- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector和- (void)forwardInvocation:(NSInvocation *)anInvocation

六、Method-Swizzling

+ (void)load{
    //获取test方法
    Method test = class_getInstanceMethod(self, @selector(test));
    //获取otherTest方法
    Method otherTest = class_getInstanceMethod(self, @selector(otherTest));
    //交换两个方法
    method_exchangeImplementations(test, otherTest);
}

- (void)test{
    NSLog(@"test");
}
- (void)otherTest{
    //实际上是调用test具体实现
    [self otherTest];
    NSLog(@"otherTest");
}

七、动态添加方法

问题:是否使用过performSelector: 方法?
答:

void testImp (void) {
    NSLog(@"test invoke");
}

+ (BOOL)resolveInstanceMethod:(SEL)sel {
    // 如果是test方法 打印日志
    if (sel == @selector(test)) {
        NSLog(@"resolveInstanceMethod:");
        // 动态添加test方法的实现
        class_addMethod(self, @selector(test), testImp, "v@:");
        //解决了实例方法调用 返回YES
        return YES;
    }else{
        // 返回父类的默认调用
        return [super resolveInstanceMethod:sel];
    }
}

八、动态方法解析

@dynamic (问题:是否使用过@dynamic 关键字?)

  • 动态运行时语言将函数决议推迟到运行时
    (当把属性标识为@dynamic时, 代表着不需要编译器在编译时为属性生成get方法和set方法的具体实现,而是在运行时具体调用get方法或者set方法时,再去添加具体实现)
  • 编译时语言在编译期进行函数决议
    (在编译期就确定了方法函数体是哪个, 具体运行过程中不能修改)

Runtime面试问题总结:

问题: [obj foo] 和 objc_msgSend()函数之间有什么关系?
答: 实际上消息传递, 在编译期处理过程后, [obj foo] 就转变成了objc_magSend(obj, @selector(foo)) , 之后开始runtime消息传递过程

问题:runtime如何通过Selector找到对应的IMP地址的?
答:考察消息传递机制.

  • 首先查找当前实例所对应类对象的缓存是否有Selector对应缓存的IMP实现, 如果缓存命中了,就把命中缓存函数返回给调用方.
  • 如果缓存没有命中,根据当前类方法列表查找Selector对应的IMP实现
  • 如果当前类没有命中, 在根据当前类superclass指针逐级查找父类方法列表,然后查找Selector对应的IMP实现.

问题:能否向编译后的类中添加实例变量?
答:(两个点 编译后的类,还是动态添加的类?)
不能.
由于runtime是支持在运行时动态添加类, 编译之前创建的类,已经完成了实例变量的布局, runtime数据结构中 class_ro_t 编译后没有办法修改的.

问题:能否向动态添加的类中添加实例变量?
答:可以. 动态添加的类调用注册类方法前,完成实例变量的添加是可以实现的.

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

推荐阅读更多精彩内容