runtime 理解

runtime是一套可以进行一些非常底层的用OC无法实现的操作的纯C语言的API。

点击进入官方文档

Objective-C 类由一个指向 objc_class 结构体的指针来表示

1
2
// An opaque type that represents an Objective-C class.
typedef struct objc_class *Class;

objc/runtime.h 中 objc_class 结构体的定义如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
struct objc_class {
Class isa OBJC_ISA_AVAILABILITY;

#if !__OBJC2__
Class super_class OBJC2_UNAVAILABLE;
const char *name OBJC2_UNAVAILABLE;//类名
long version OBJC2_UNAVAILABLE;
long info OBJC2_UNAVAILABLE;
long instance_size OBJC2_UNAVAILABLE;
struct objc_ivar_list *ivars OBJC2_UNAVAILABLE;//该类的成员变量列表
struct objc_method_list **methodLists OBJC2_UNAVAILABLE;
struct objc_cache *cache OBJC2_UNAVAILABLE;//缓存最近使用的方法
struct objc_protocol_list *protocols OBJC2_UNAVAILABLE;
#endif

} OBJC2_UNAVAILABLE;

一个类的实例由结构体 objc_object 来表示

1
2
3
4
5
6
7
// Represents an instance of a class.
struct objc_object {
Class isa OBJC_ISA_AVAILABILITY;
};

// A pointer to an instance of a class.
typedef struct objc_object *id;

当我们向一个 Objective-C 对象发送消息时,运行时库会根据实例对象的 isa 指针找到这个实例对象所属的类。Runtime 库会在类或其父类的方法列表中查询并执行与消息对应的 selector 指向的方法。

元类(MetaClass)是一个类对象的类

所有的类自身也是一个对象,我们可以向这个对象发送消息 —— 调用类方法。

meta-class 也是一个类,也可以向它发送一个消息。Objective-C 的设计者让所有的 meta-class 的 isa 指向基类的 meta-class,以此作为它们的所属类。即,任何 NSObject 继承体系下的 meta-class 都使用 NSObject 的 meta-class 作为自己的所属类,而基类的 meta-class 的 isa 指针是指向它自己。这样就形成了一个完美的闭环。

1
2
3
4
5
6
7
8
9
10
11
12
UIView *testView = [[UIView alloc] init];

NSLog(@"%p", self);
NSLog(@"%s", class_getName([testView class] ));
NSLog(@"class is %s, superclass is %s", class_getName([self class]),
class_getName([self superclass]));

Class currentClass = [self class];
for (int i = 0; i < 3; i++) {
NSLog(@"Following the isa pointer %d times gives %p", i, currentClass);
currentClass = objc_getClass((__bridge void *)currentClass);
}

控制台输出:

1
2
3
4
5
6
0x7fa410c1f990
UIView
class is ViewController, superclass is UIViewController
Following the isa pointer 0 times gives 0x101c4e6c0
Following the isa pointer 1 times gives 0x0
Following the isa pointer 2 times gives 0x0

runtime 对类和对象的操作

runtime 提供了大量的函数来操作类与对象。类的操作方法大部分是以 class_ 为前缀的,而对象的操作方法大部分是以 objc_ 或 object_ 为前缀。

对类操作

runtime 提供的操作类的方法主要是针对结构体 objc_class 中的各个字段的。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
// Returns the name of a class.
const char * class_getName(Class cls);
// Returns the superclass of a class.
Class class_getSuperclass(Class cls);
// Sets the superclass of a given class, return the old superclass for cls.
Class class_setSuperclass(Class cls, Class newSuper);
// Returns a Boolean value that indicates whether a class object is a metaclass.
BOOL class_isMetaClass(Class cls);
// Returns the size of instances of a class.
size_t class_getInstanceSize(Class cls);
// Returns the Ivar for a specified instance variable of a given class.
Ivar class_getInstanceVariable(Class cls, const char* name);
// Returns the Ivar for a specified class variable of a given class.
Ivar class_getClassVariable(Class cls, const char* name);
// Returns a description of the Ivar layout for a given class.
const char *class_getIvarLayout(Class cls);

...