runTime:Objective-C是基于C语言加入了面向对象特性和消息转发机制的动态语言,这意味着它不仅需要一个编译器,还需要Runtime系统来动态创建类和对象,进行消息发送和转发。
1. 什么是运行时
1> 运行时(runtime)是一套纯C语言的API(纯C语言的库,包含了很多功能强大的C语言函数)。
2> 我们平时写的OC代码,在程序'运行'期间,其实最终都会被转化为'runtime的C语言代码'
总之,'调用方法本质上就是发送消息',即 先给 'NSObject' 类发送 'alloc' 消息,创建出一个对象;再给创建出来的对象发送 'init' 消息,对对象进行初始化操作。
2. 运行时的应用场景
runtime属于OC的底层,利用runtime可以做一些非常底层的操作,比如:
1> 动态给对象添加成员变量和成员方法
2> 动态交换两个方法的实现(特别是'交换系统自带的方法')
3> 动态获得某个类的所有成员变量、成员方法
3. 如何应用运行时?
1> 将某些OC代码转为运行时代码:探究底层,比如block的实现原理
2> 拦截系统自带的方法调用,比如拦截imageNamed:、viewDidLoad、alloc
3> 实现字典和模型的自动转换
4> 实现分类也可以增加属性(每个对象的属性互不干扰)
5> 实现NSCoding的自动归档和自动解档
4. 运行时常用的函数
1>
*Methodclass_getClassMethod(Class cls, SEL name)获得某个类的类方法
* Methodclass_getInstanceMethod(Class cls, SEL name)获得某个类的对象方法
Ivar * class_copyIvarList :获得某个类内部的所有成员变量
Method * class_copyMethodList :获得某个类内部的所有方法
*voidmethod_exchangeImplementations(Method m1, Method m2) 交换2个方法的实现
*voidobjc_setAssociatedObject(idobject,constvoid*key, idvalue, objc_AssociationPolicy policy)将值value跟对象object关联起来(将值value存储到对象object中)参数key:将来可以通过key取出这个存储的值参数policy:存储策略(assign、copy、retain)
* idobjc_getAssociatedObject(idobject,constvoid*key)利用参数key将对象object中存储的对应值取出来* Ivar *class_copyIvarList(Class cls, unsignedint*outCount) 获得某个类的所有成员变量(outCount会返回成员变量的总数)*constchar*ivar_getName(Ivar v)获得成员变量的名字
*constchar*ivar_getTypeEncoding(Ivar v) 获得成员变量的类型*voidfree(void*);释放内存(当C语言函数名中包含了copy、create、retain、new等词语,那么就需要在最后释放资源)
2>
*voidobjc_msgSend(void)给某个对象发送某个消息
对象保存在堆区,包括共有属性,和方法的声名. 和 isa指针,
person 对象 调用 message 这个方法的时候, 并不会把方法的声名和实现绑定到一起, 而是在运行的时候,根据对象的 isa 指针找到对应的类,然后在 cecha 中对应的方法, 如果找不到,会在消息链表中找方法对应的实现, 如果还是找不到的话,会更具 superClass 在父类找, 直到找为止,
cecha 由于类的很多方法不是经常调用的, 系统会把经常的调用的方法的实现缓存起来, 优化性能.\
现在让我们看一下objc_msgSend它具体是如何发送消息:
首先根据receiver对象的isa指针获取它对应的class
优先在class的cache查找message方法,如果找不到,再到methodLists查找
如果没有在class找到,再到super_class查找
一旦找到message这个方法,就执行它实现的IMP。