一、什么是runtime
简而言之,Objective-C
Runtime
是一个将C
语言转化为面向对象语言的扩展。
我们将C++
和Objective
进行对比,虽然C++
和Objective-C
都是在C
的基础上加入面向对象的特性扩充而成的程序设计语言,但二者实现的机制差异很大。C++
是基于静态类型,而Objective-C
是基于动态运行时类型。也就是说用C++
编写的程序编译时就直接编译成了可令机器读懂的机器语言;用Objective-C
编写的程序不能直接编译成可令机器读懂的机器语言,而是在程序运行的时候,通过Runtime
把程序转为可令机器读懂的机器语言。Runtime
是Objective
不可缺少的重要一部分。
runtime
是一个c
和汇编
写的动态库,它就像一个小小的系统,将OC和C紧密关联,这个系统主要做两件事 :
- 1、封装C语言的结构体和函数,让开发者在运行时创建、检查或者修改类、对象和方法等等。
- 2、传递消息,找出方法的最终执行代码。
- 1、无参数
OC定义方法
[receiver message]
runtime转化C语言的代码:
objc_msgSend(receiver, selector)
- 2、有参数
OC定义方法
[receiver message:arg1 ...]
runtime转化C语言的代码:
objc_msgSend(receiver, selector, arg1, arg2, ...)
- 3、多方法
OC定义方法
NSObject *object = [[NSObject alloc] init];
runtime转化C语言的代码:
objc_msgSend(objc_msgSend([NSObject class],@selector(alloc)),@selector(init));
二、与runtime交互
bjc
从三种不同的层级上与 Runtime
系统进行交互,分别是:
- 通过
Objective-C
源代码 - 通过
Foundation
框架的NSObject
类定义的方法 - 通过对
runtime
函数的直接调用。
1、Objective-C源代码
编写OC
代码,程序在运行时,runtime
会自动将OC
转化成C
语言代码。
2、NSObject的方法
Cocoa
中大多数类都继承于NSObject
类,也就自然继承了它的方法。最特殊的例外是NSProxy
,它是个抽象超类,它实现了一些消息转发有关的方法,可以通过继承它来实现一个其他类的替身类或是虚拟出一个不存在的类。
有的NSObject
中的方法起到了抽象接口的作用,比如description
方法需要你重载它并为你定义的类提供描述内容。NSObject
还有些方法能在运行时获得类的信息,并检查一些特性,比如class返回对象的类;isKindOfClass:
和isMemberOfClass:
则检查对象是否在指定的类继承体系中;respondsToSelector:
检查对象能否响应指定的消息;conformsToProtocol:
检查对象是否实现了指定协议类的方法;methodForSelector:
则返回指定方法实现的地址。
3、Runtime的函数
Runtime
系统是一个由一系列函数和数据结构组成,具有公共接口的动态共享库。头文件存放于/usr/include/objc
目录下。许多函数允许你用纯C
代码来重复实现 Objc
中同样的功能。虽然有一些方法构成了NSObject
类的基础,但是你在写 Objc
代码时一般不会直接用到这些函数的,除非是写一些 Objc
与其他语言的桥接或是底层的debug
工作。
三、Runtime元素认知
objc_msgSend:方法,它的真身是这样的:
id objc_msgSend ( id self, SEL op, ... );
1、SEL(objc.h)
objc_msgSend
函数第二个参数类型为SEL
,它是selector
在Objc
中的表示类型(Swift
中是Selector
类)。selector
是方法选择器,可以理解为区分方法的 ID
,而这个 ID
的数据结构是SEL
:
typedef struct objc_selector *SEL;
objc_selector的定义如下:
struct objc_selector {
char *name; OBJC2_UNAVAILABLE;// 名称
char *types; OBJC2_UNAVAILABLE;// 类型
};
name
和types
都是char
类型。
其实它就是个映射到方法的C
字符串,你可以用 Objc
编译器命令@selector()
或者 Runtime
系统的sel_registerName
函数来获得一个SEL
类型的方法选择器。
不同类中相同名字的方法所对应的方法选择器是相同的,即使方法名字相同而变量类型不同也会导致它们具有相同的方法选择器,于是 Objc
中方法命名有时会带上参数类型;
2、id(objc.h)
objc_msgSend
第一个参数类型为id
,它是一个指向类实例的指针:
typedef struct objc_object *id;
objc_object
定义如下:
struct objc_object {
Class isa OBJC_ISA_AVAILABILITY;
};
- isa:
objc_object
结构体包含一个isa
指针,根据isa
指针就可以找到对象所属的类。objc_object
(实例对象)中isa
指针指向的类结构称为class
(也就是该对象所属的类)其中存放着普通成员变量与动态方法(“-”开头的方法);
当我们向一个
Objective-C
对象发送消息时,运行时库会根据实例对象的isa
指针找到这个实例对象所属的类。Runtime
库会在类的方法列表及父类的方法列表中去寻找与消息对应的selector
指向的方法。找到后即运行这个方法。
当创建一个特定类的实例对象时,分配的内存包含一个
objc_object
数据结构,然后是类的实例变量的数据。NSObject
类的alloc
和allocWithZone:
方法使用函数class_createInstance
来创建objc_object
数据结构。
另外还有我们常见的id
,它是一个objc_object
结构类型的指针。它的存在可以让我们实现类似于C++
中泛型的一些操作。该类型的对象可以转换为任何一种对象,有点类似于C
语言中void *
指针类型的作用。
PS:
isa
指针不总是指向实例对象所属的类,不能依靠它来确定类型,而是应该用class
方法来确定实例对象的类。因为KVO
的实现机理就是将被观察对象的isa
指针指向一个中间类而不是真实的类,这是一种叫做isa-swizzling
的技术;
3、Class(objc.h,runtime.h)
之所以说isa
是指针是因为Class
其实是一个指向objc_class
结构体的指针:
typedef struct objc_class *Class;
objc_class
在runtime.h
定义如下:
struct objc_class {
Class _Nonnull isa OBJC_ISA_AVAILABILITY;
#if !__OBJC2__
Class _Nullable super_class OBJC2_UNAVAILABLE;// 父类
const char * _Nonnull name OBJC2_UNAVAILABLE;// 类名
long version OBJC2_UNAVAILABLE;// 类的版本信息,默认为0,可以通过runtime函数class_setVersion或者class_getVersion进行修改、读取
long info OBJC2_UNAVAILABLE;;// 类信息,供运行时期使用的一些位标识,如CLS_CLASS (0x1L) 表示该类为普通 class,其中包含实例方法和变量;CLS_META (0x2L) 表示该类为 metaclass,其中包含类方法;
long instance_size OBJC2_UNAVAILABLE;// 该类的实例变量大小(包括从父类继承下来的实例变量)
struct objc_ivar_list * _Nullable ivars OBJC2_UNAVAILABLE;// 该类的成员变量地址列表
struct objc_method_list * _Nullable * _Nullable methodLists OBJC2_UNAVAILABLE;// 方法地址列表,与 info 的一些标志位有关,如CLS_CLASS (0x1L),则存储实例方法,如CLS_META (0x2L),则存储类方法;
struct objc_cache * _Nonnull cache OBJC2_UNAVAILABLE;// 缓存最近使用的方法地址,用于提升效率;
struct objc_protocol_list * _Nullable protocols OBJC2_UNAVAILABLE;// 存储该类声明遵守的协议的列表
#endif
} OBJC2_UNAVAILABLE;
PS:
OBJC2_UNAVAILABLE
之类的宏定义是苹果在Objc
中对系统运行版本进行约束的黑魔法,为的是兼容非Objective-C
2.0的遗留逻辑,但我们仍能从中获得一些有价值的信息,有兴趣的可以查看源代码。
-
(1)、
isa
:此处isa
指针指向的类结构称为metaclass
,其中存放着static
类型的成员变量与static
类型的方法(“+”开头的方法)。 -
(2)、
super_class
: 指向该类的父类的指针,如果该类是根类(如NSObject
或NSProxy
),那么super_class
就为nil
。 -
(3)、
cache
:用于缓存最近使用的方法。一个接收者对象接收到一个消息时,它会根据isa
指针去查找能够响应这个消息的对象。在实际使用中,这个对象只有一部分方法是常用的,很多方法其实很少用或者根本用不上。这种情况下,如果每次消息来时,我们都是methodLists
中遍历一遍,性能势必很差。这时,cache
就派上用场了。在我们每次调用过一个方法后,这个方法就会被缓存到cache
列表中,下次调用的时候runtime
就会优先去cache
中查找,如果cache
没有,才去methodLists
中查找方法。这样,对于那些经常用到的方法的调用,但提高了调用的效率。
针对cache
,我们用下面例子来说明其执行过程:NSArray *array = [[NSArray alloc] init];
其流程是:
- ①、
[NSArray alloc]
先被执行。因为NSArray
没有+alloc
方法,于是去父类NSObject
去查找。 - ②、检测
NSObject
是否响应+alloc
方法,发现响应,于是检测NSArray
类,并根据其所需的内存空间大小开始分配内存空间,然后把isa
指针指向NSArray
类。同时,+alloc
也被加进cache
列表里面。 - ③、接着,执行
-init
方法,如果NSArray
响应该方法,则直接将其加入cache
;如果不响应,则去父类查找。 - ④、在后期的操作中,如果再以
[[NSArray alloc] init]
这种方式来创建数组,则会直接从cache
中取出相应的方法,直接调用。
- ①、
-
(4)、
version
:我们可以使用这个字段来提供类的版本信息。这对于对象的序列化非常有用,它可是让我们识别出不同类定义版本中实例变量布局的改变。
总结:类与对象的区别就是类比对象多了很多特征成员,类也可以当做一个objc_object
来对待,也就是说类和对象都是对象,分别称作类对象(class object
)和实例对象(instance object
),这样我们就可以区别对象和类了。
为了处理类和对象的关系,runtime
库创建了一种叫做元类 (Meta Class
) 的东西,类对象所属类型就叫做元类,它用来表述类对象本身所具备的元数据。类方法就定义于此处,因为这些方法可以理解成类对象的实例方法。每个类仅有一个类对象,而每个类对象仅有一个与之相关的元类。当你发出一个类似[NSObject alloc]
的消息时,你事实上是把这个消息发给了一个类对象 (Class Object
) ,这个类对象必须是一个元类的实例,而这个元类同时也是一个根元类 (root meta class
) 的实例。所有的元类最终都指向根元类为其超类。所有的元类的方法列表都有能够响应消息的类方法。所以当 [NSObject alloc]
这条消息发给类对象的时候。
PS:在
objc_class
结构体中:ivars
是objc_ivar_list
指针;methodLists
是指向objc_method_list
指针的指针。也就是说可以动态修改*methodLists
的值来添加成员方法,这也是Category
实现的原理,同样解释了Category
不能添加属性的原因。
4、IMP(objc.h)
IMP在objc.h中的定义是:
#if !OBJC_OLD_DISPATCH_PROTOTYPES
typedef void (*IMP)(void /* id, SEL, ... */ );
#else
typedef id (*IMP)(id, SEL, ...);
#endif
IMP
是“implementation
”的缩写,它是由编译器生成的一个函数指针。当你发起一个ObjC
消息后,这个函数指针决定了最终执行哪段代码。
在iOS
的Runtime
中,Method
通过selector
和IMP
两个属性,实现了快速查找方法及实现,相对提高了性能,又保持了灵活性。
5、Method(runtime.h)
Method代表类中的某个方法的类型。
/// An opaque type that represents a method in a class definition.
typedef struct objc_method *Method;
objc_method
的定义如下:
struct objc_method {
SEL method_name OBJC2_UNAVAILABLE;// 方法名
char *method_types OBJC2_UNAVAILABLE;// 方法类型
IMP method_imp OBJC2_UNAVAILABLE;// 方法实现
}OBJC2_UNAVAILABLE;
- (1)、
method_name
(方法名):类型为SEL
,前面提到过相同名字的方法即使在不同类中定义,它们的方法选择器也相同; - (2)、
method_types
(方法类型):是个char
指针,其实存储着方法的参数类型和返回值类型; - (3)、
method_imp
(方法实现):指向了方法的实现,本质上是一个函数指针。
6、Ivar(runtime.h)
Ivar
代表类中实例变量的类型
/// An opaque type that represents an instance variable.
typedef struct objc_ivar *Ivar;
objc_ivar
的定义如下:
struct objc_ivar {
char *ivar_name OBJC2_UNAVAILABLE;// 变量名
char *ivar_type OBJC2_UNAVAILABLE;// 变量类型
int ivar_offset OBJC2_UNAVAILABLE;// 基地址偏移字节
#ifdef __LP64__
int space OBJC2_UNAVAILABLE;// 占用空间
#endif
}
7、Category(runtime.h)
Category
的定义如下:
/// An opaque type that represents a category.
typedef struct objc_category *Category;
objc_category
的定义如下:
struct objc_category {
char *category_name OBJC2_UNAVAILABLE;// 类别名称
char *class_name OBJC2_UNAVAILABLE;// 类名
struct objc_method_list *instance_methods OBJC2_UNAVAILABLE;// 实例方法列表
struct objc_method_list *class_methods OBJC2_UNAVAILABLE;// 类方法列表
struct objc_protocol_list *protocols OBJC2_UNAVAILABLE;// 协议列表
}
8、objc_property_t(runtime.h)
objc_property
是类中的属性,它的定义如下
/// An opaque type that represents an Objective-C declared property.
typedef struct objc_property *objc_property_t;
objc_property
是内置的类型,与之关联的还有一个objc_property_attribute_t
,它是属性的attribute
,也就是其实是对属性的详细描述,包括属性名称、属性编码类型、原子类型/非原子类型等。它的定义如下:
/// Defines a property attribute
typedef struct {
const char *name; // 名称
const char *value; // 值(通常是空的)
} objc_property_attribute_t;
9、Cache(runtime.h)
- 用于快速查找方法执行函数
- 是可增量扩展的哈希表结构
- 是局部性原理的最佳应用
Catch
的定义如下:
typedef struct objc_cache *Cache OBJC2_UNAVAILABLE;
objc_cache
的定义如下
struct objc_cache {
unsigned int mask /* total = mask + 1 */ OBJC2_UNAVAILABLE;
unsigned int occupied OBJC2_UNAVAILABLE;
Method buckets[1] OBJC2_UNAVAILABLE;
};
- (1)、
mask
:指定分配cache buckets
的总数。在方法查找中,Runtime
使用这个字段确定数组的索引位置。 - (2)、
occupied
:实际占用cache buckets
的总数。 - (3)、
buckets
:指定Method
数据结构指针的数组。这个数组可能包含不超过mask+1个元素。需要注意的是,指针可能是NULL
,表示这个缓存bucket
没有被占用,另外被占用的bucket
可能是不连续的。这个数组可能会随着时间而增长。
objc_msgSend
每调用一次方法后,就会把该方法缓存到cache
列表中,下次的时候,就直接优先从cache
列表中寻找,如果cache
没有,才从methodLists
中查找方法。