iOS知识点

iOS-RunTime相关知识点整理

2019-01-21  本文已影响0人  木子奕

Runtime.png

Runtime 内容汇总

904629-3ab2a221da9dae6b.png

这节的讲解内容


数据结构

image.png image.png image.png

cache_t存储的是类的缓存快速查找的hash表

image.png image.png

class_rw_t主要存储的是和分类相关的内容

image.png

class_ro_t主要存储的是和原始类相关的内容

image.png

method_t的结构体与函数的对应关系如下所示

image.png

方法的Type Encondings

image.png

类的整体存储结构

904629-82cd4639a024707a.png

上图可以看出 Runtime 整体的数据结构


类对象与元类对象 & 消息传递

类对象与元类对象

看一个官方很经典的图

904629-b0909bd8eaa595a0.png

objc_class 继承于 objc_object,也就是说一个 ObjC 类本身同时也是一个对象,为了处理类和对象的关系,runtime库创建了一种叫做元类 (Meta Class)的东西,类对象所属类型就叫做元类,它用来表述类对象本身所具备的元数据。类方法就定义于此处,因为这些方法可以理解成类对象的实例方法。每个类仅有一个类对象,而每个类对象仅有一个与之相关的元类。当你发出一个类似[NSObject alloc] 的消息时,你事实上是把这个消息发给了一个类对象 (Class Object),这个类对象必须是一个元类的实例,而这个元类同时也是一个根元类 (root meta class)的实例。所有的元类最终都指向根元类为其超类。所有的元类的方法列表都有能够响应消息的类方法。所以当 [NSObject alloc]这条消息发给类对象的时候,objc_msgSend() 会去它的元类里面去查找能够响应消息的方法,如果找到了,然后对这个类对象执行方法调用。

上图实线是 superclass 指针,虚线是isa指针。 有趣的是根元类的超类是 NSObject,而 isa 指向了自己,而 NSObject 的超类为nil,也就是它没有超类。

这里讲的非常详细了

消息传递
904629-6914043db2c075c2.png

这就是消息传递的一个流程,首先查缓存,无缓存,查方法列表,依然没命中,再顺次查找各个父类方法列表,如果都没有名字,就转到消息转发流程

如果问具体在缓存和方法列表查找过程中的内部情况,可以简单的这样答一下

看一个经典的面试题
904629-0f9525b131902843.png

结果是都打印出来 phone

为什么呢?先看一下 [self class][super class]的消息传递

904629-40e52989b6b43445.png 904629-05f1dff5c55559df.png

这里可以看到 objc_super 结构体当中 receiver 就是当前对象 。 也就是说 无论是 [self class]或者 [superclass] ,接收者都是当前对象!!

继续, 我们对应着官方流程图(图三)来解答一下

当前的实例是 phone [self class] -->objc_msgSend(self,@selector(class)), 首先通过实例的 isa 指针找到 phone的类对象,它本身是没有 class 方法的,然后往父类找,也没有。 顺次一直查找到根类也就是 NSObject ,将class的具体实现返回给调用方,也就是 phone

instance of Subclass --> Subclass(class) -->Superclass(class)-->Root class(class)

[super class] -->objc_msgSendSuper(self,@selector(class))。 接受者依然是当前的 phone这个实例,区别不同的是 它会从父类的方法了列表开始查找,也就是

instance of Subclass --> Superclass(class)-->Root class(class)

所以依然还是 phone;

方法的查找过程

image.png image.png

消息转发

904629-43edc5597d2c72ea.png

实践出真知,跑一下demo


@dynamic getter和setter 在运行时生成

image.png
上一篇 下一篇

猜你喜欢

热点阅读