[iOS]从拾遗到Runtime(上)
文章目录
- [iOS]从拾遗到Runtime(上)
- 写在前面
- 名词介绍
- instance 实例对象
- class 类对象
- meta-class 元类对象
- 为什么要有元类?
- runtime
- Method(objc_method)
- SEL(objc_selector)
- IMP
- 类缓存(objc_cache)
- Category(objc_category)
- 消息传递
- 消息传递的流程
- 消息转发
- 动态方法解析
- 备用接收者
- 完整消息转发
- 参考博客
写在前面
最近看到学弟学习isMemberOfClass方法和isKindOfClass方法时遇到的一个问题
[[NSString class] isMemberOfClass:[NSObject class]];
[NSString isMemberOfClass:[NSObject class]];
这两句有啥区别
我知道第二句是类方法 第一句呢 和第二句啥区别 也是类方法吗
我刚想说 看我博客去
但是想了想我的博客真的惨不忍睹
当时写的时候语焉不详 再加上我对这一块也没啥印象了
于是问问gpt吧
笨蛋gpt信誓旦旦告诉我 类对象就是实例
好 问来问去不如自己动手 毕竟高中就教过实践是检验真理的唯一标准
NSLog(@"%@", [[NSString class] stringWithFormat:@"123"]);
NSLog(@"%d", [NSString isEqual:[NSString class]]);
打印结果
第一句是让[NSString class]的返回值再调用NSString的类方法 成功
第二句是用isEqual方法比较俩者是否意义相同 成功
那么目前来看 好像可以把二者作用画等号了
但这还不够
class方法返回值是类对象
到底啥是类对象 实例 元类?
我们说 类方法实例方法 类对象实例对象
那类对象就是实例的说法显然不太合理
再加上消息转发那部分先前也是写的匆匆忙忙
不如在这一篇runtime一次性讲清楚(`ヮ´ )
名词介绍
instance 实例对象
instance对象就是通过alloc方法创建出来的对象,每次调用alloc方法都会生成新的instance对象
instance对象在内存中存放的信息包括
- isa指针
- 其他成员变量
/// 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;
class 类对象
class对象的作用是用来描述一个instance对象,它内部存放一个类的属性信息(@property)、对象方法信息(instance method)、协议信息(protocol)、成员变量信息(ivar),另外class对象里面还有两个指针,isa指针 和 superclass指针。
Objective-C类是由Class类型来表示的,它实际上是一个指向objc_class
结构体的指针
typedef struct objc_class *Class;
查看objc/runtime.h中objc_class结构体的定义如下:
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;long info OBJC2_UNAVAILABLE;long instance_size OBJC2_UNAVAILABLE;struct objc_ivar_list * _Nullable ivars OBJC2_UNAVAILABLE;struct objc_method_list * _Nullable * _Nullable methodLists OBJC2_UNAVAILABLE;struct objc_cache * _Nonnull cache OBJC2_UNAVAILABLE;struct objc_protocol_list * _Nullable protocols OBJC2_UNAVAILABLE;
#endif} OBJC2_UNAVAILABLE;
类对象就是一个结构体struct objc_class,这个结构体存放的数据称为元数据(metadata),
该结构体的第一个成员变量也是isa指针,这就说明了Class本身其实也是一个对象,因此我们称之为类对象
类对象在编译期产生用于创建实例对象,是单例
meta-class 元类对象
meta-class对象的作用是用来描述一个class对象
跟class一样,元类对象在内存中也是只有一份的
它内部存储了 isa指针 + superclass指针 + 类方法信息(+方法)
类对象中的元数据存储的都是如何创建一个实例的相关信息
那么类对象和类方法应该从哪里创建呢? 就是从isa指针指向的结构体创建
类对象的isa指针指向的我们称之为元类(metaclass), 元类中保存了创建类对象以及类方法所需的所有信息
来看这张很经典的图
通过上图我们可以看出整个体系构成了一个自闭环,struct objc_object结构体实例它的isa指针指向类对象
类对象的isa指针指向了元类,super_class指针指向了父类的类对象
而元类的super_class指针指向了父类的元类,那元类的isa指针又指向了自己
为什么要有元类?
元类(Meta Class)是一个类对象的类。在上面我们提到,所有的类自身也是一个对象,我们可以向这个对象发送消息(即调用类方法)。为了调用类方法,这个类的isa指针必须指向一个包含这些类方法的一个objc_class结构体。这就引出了meta-class的概念,元类中保存了创建类对象以及类方法所需的所有信息。任何NSObject继承体系下的meta-class都使用NSObject的meta-class作为自己的所属类,而基类的meta-class的isa指针是指向它自己
runtime
因为Objective-C是一门动态语言,所以它将一些决策工作从编译、连接过程推迟到运行时。所以只有编译器是不够的,还需要一个运行时系统 (runtime system) 来执行编译后的代码。这就是 Objective-C Runtime 系统存在的意义,它是整个Objective-C运行框架的一块基石
通俗来说
OC 是一门动态语言,函数调用变成了消息发送,在编译期不能知道要调用哪个函数。所以 Runtime 无非就是去解决如何在运行时期找到调用方法这样的问题
Method(objc_method)
Method和我们平时理解的函数是一致的,就是表示能够独立完成一个功能的一段代码
来看定义
runtime.h
/// An opaque type that represents a method in a class definition.代表类定义中一个方法的不透明类型
typedef struct objc_method *Method;
struct objc_method {SEL method_name OBJC2_UNAVAILABLE;char *method_types OBJC2_UNAVAILABLE;IMP method_imp OBJC2_UNAVAILABLE;
在这个结构体中能看到SEL和IMP,说明SEL和IMP其实都是Method的属性
SEL 与 IMP 的关系非常类似于 HashTable 中 key 与 value 的关系
OC中不支持函数重载的原因就是因为一个类的方法列表中不能存在两个相同的 SEL
但是多个方法却可以在不同的类中有一个相同的SEL
不同类的实例对象执行相同的 SEL 时
会在各自的方法列表中去根据 SEL 去寻找自己对应的IMP
这使得OC可以支持函数重写
在iOS的Runtime中,Method通过selector和IMP两个属性,实现了快速查询方法及实现,相对提高了性能,又保持了灵活性。
SEL(objc_selector)
objc_msgSend函数第二个参数类型为SEL,它是selector在Objective-C中的表示类型(Swift中是Selector类)。selector是方法选择器,可以理解为区分方法的 ID,而这个 ID 的数据结构是SEL:
选择器 好名字不是吗
眼尖记性好的观众可能记着之前消息转发讲过选择子
其实一个意思
顺带看看那里
Objc.h
/// An opaque type that represents a method selector.代表一个方法的不透明类型
typedef struct objc_selector *SEL;
然后
@property SEL selector;
相信大家不难看出selector是SEL的一个实例
其实selector就是个映射到方法的C字符串,你可以用 Objective-C 编译器命令@selector()或者 Runtime 系统的sel_registerName函数来获得一个 SEL 类型的方法选择器。
selector既然是一个string,我觉得应该是类似className+method的组合,命名规则有两条:
- 同一个类,selector不能重复
- 不同的类,selector可以重复
这也带来了一个弊端,我们在写C代码的时候,经常会用到函数重载,就是函数名相同,参数不同,但是这在Objective-C中是行不通的,因为selector只记了method的name,没有参数,所以没法区分不同的method。
- (void)caculate(NSInteger)num;
- (void)caculate(CGFloat)num;
这种就会报错
在不同类中相同名字的方法所对应的方法选择器是相同的
即使方法名字相同而变量类型不同也会导致它们具有相同的方法选择器
IMP
IMP就是指向最终实现程序的内存地址的指针
/// A pointer to the function of a method implementation. 指向一个方法实现的指针
typedef id (*IMP)(id, SEL, ...);
#endif
类缓存(objc_cache)
当Objective-C运行时通过跟踪它的isa指针检查对象时,它可以找到一个实现许多方法的对象
然而,你可能只调用它们的一小部分,并且每次查找时,搜索所有选择器的类分派表没有意义
所以类实现一个缓存,每当你搜索一个类分派表,并找到相应的选择器,它把它放入它的缓存
所以当objc_msgSend查找一个类的选择器,它首先搜索类缓存
这是基于这样的理论:如果你在类上调用一个消息,你可能以后再次调用该消息
为了加速消息分发, 系统会对方法和对应的地址进行缓存,就放在上述的objc_cache
所以在实际运行中,大部分常用的方法都是会被缓存起来的
Runtime系统实际上非常快,接近直接执行内存地址的程序速度
Category(objc_category)
Category是表示一个指向分类的结构体的指针
来看
struct category_t {
// name:是指 class_name 而不是 category_nameconst char *name;
// cls:要扩展的类对象,编译期间是不会定义的,而是在Runtime阶段通过name对 应到对应的类对象classref_t cls;
// instanceMethods:category中所有给类添加的实例方法的列表struct method_list_t *instanceMethods;
// classMethods:category中所有添加的类方法的列表 struct method_list_t *classMethods;
// protocols:category实现的所有协议的列表struct protocol_list_t *protocols;
// instanceProperties:表示Category里所有的properties,这就是我们可以通过objc_setAssociatedObject和objc_getAssociatedObject增加实例变量的原因,不过这个和一般的实例变量是不一样的struct property_list_t *instanceProperties;
};
从上面的category_t的结构体中可以看出,分类中可以添加实例方法,类方法,甚至可以实现协议,添加属性,不可以添加成员变量
消息传递
消息传递的流程
Objective-C中所有方法的调用/类的生成都在运行时进行,我们可以通过类名/方法名反射得到相应的类和方法,也可以替换某个类的方法为新的实现,理论上你可以在运行时通过类名/方法名调用到任意Objective-C 方法,替换任何类的实现以及新增任意类
比方说我们写一个调用方法[receiver foo]
首先这行代码会被改写成objc_msgSend(self, _cmd);
这是一个runtime的函数
其原型如下
// 第一个参数类型是发送者, 第二个参数类型是SEL。SEL在OC中是selector方法选择器
id objc_msgSend ( id _Nullable self, SEL op, ... );
实际上,我们在调用的方法的过程,其实在Runtime中就是消息发送
objc_msgSend的实现是由汇编语言实现,根据CPU架构实现的过程各不相同
objc_msgSend会做以下几件事情:
-
检测这个selector是不是要忽略 检查target是不是为nil
-
如果这里有相应的nil的处理函数,就跳转到相应的函数中
如果没有处理nil的函数,就自动清理并返回(这一点就是为何在Objective-C中给nil发送消息不会崩溃的原因) -
确定不是给nil发消息之后,在该对象的类(Class)的缓存中查找方法对应的IMP (俗称快查)
-
如果找到,就跳转进去执行;
如果没有找到,执行下一步; -
在方法列表中继续查找,一直找到NSObject为止;(俗称慢找)
如果还没有找到,那就需要开始消息转发阶段了。至此,发送消息Messaging阶段完成。这一阶段主要完成的是通过select()快速查找IMP的过程
self与_cmd是两个编译器会自动添加的隐藏参数,self是一个指向接收对象的指针,_cmd为方法选择器。这个函数的实现为汇编版本,苹果开源的项目中共有6种对不同平台的汇编实现
这个东西有空再补
消息转发
前文介绍了进行一次发送消息会在相关的类对象中搜索方法列表,如果找不到则会沿着继承树向上一直搜索直到继承树根部(通常为NSObject),如果还是找不到并且消息转发都失败了就回执行doesNotRecognizeSelector:
方法报unrecognized selector
错。那么消息转发到底是什么呢?接下来将会逐一介绍最后的三次机会
动态方法解析
首先Objective-C运行时会调用 +resolveInstanceMethod:
或者 +resolveClassMethod:
让你有机会提供一个函数实现
如果你添加了函数并返回YES, 那运行时系统就会重新启动一次消息发送的过程
#import "ViewController.h"
#import <objc/runtime.h>@interface ViewController ()@end@implementation ViewController- (void)viewDidLoad {[super viewDidLoad];// Do any additional setup after loading the view, typically from a nib.NSString *parameter = @"Some parameter"; // 假设的参数// 执行foo函数并传递参数[self performSelector:@selector(foo:) withObject:parameter];
}+ (BOOL)resolveInstanceMethod:(SEL)sel {if (sel == @selector(foo:)) { // 如果是执行foo函数,就动态解析,指定新的IMPclass_addMethod([self class], sel, (IMP)fooMethod, "v@:@"); // 更新编码以匹配NSString参数return YES;}return [super resolveInstanceMethod:sel];
}void fooMethod(id obj, SEL _cmd, NSString *param) { // 添加参数声明NSLog(@"Doing foo with parameter: %@", param); // 使用传入的参数
}@end
打印结果
可以看到虽然没有实现foo:这个函数,但是我们通过class_addMethod动态添加fooMethod函数,并执行fooMethod这个函数的IMP。从打印结果看,成功实现了
如果resolve方法返回 NO ,运行时就会移到下一步:forwardingTargetForSelector。
备用接收者
如果目标对象实现了-forwardingTargetForSelector:,Runtime 这时就会调用这个方法,给你把这个消息转发给其他对象的机会
举例如下
#import "ViewController.h"
#import "objc/runtime.h"@interface MYPerson: NSObject@end@implementation MYPerson- (void)foo {NSLog(@"Doing foo");//MYPerson的foo函数
}@end@interface ViewController ()@end@implementation ViewController- (void)viewDidLoad {[super viewDidLoad];// Do any additional setup after loading the view, typically from a nib.//执行foo函数[self performSelector:@selector(foo)];
}+ (BOOL)resolveInstanceMethod:(SEL)sel {return YES;//返回YES,进入下一步转发
}- (id)forwardingTargetForSelector:(SEL)aSelector {if (aSelector == @selector(foo)) {return [MYPerson new];//返回MYPerson对象,让MYPerson对象接收这个消息}return [super forwardingTargetForSelector:aSelector];
}@end
结果正确
如果在这一步还不能处理未知消息,则唯一能做的就是启用完整的消息转发机制了
完整消息转发
首先它会发送-methodSignatureForSelector:消息获得函数的参数和返回值类型
-
如果-methodSignatureForSelector:返回nil
Runtime则会发出 -doesNotRecognizeSelector: 消息,程序这时也就挂掉了 -
如果返回了一个函数签名
Runtime就会创建一个NSInvocation 对象并发送 -forwardInvocation:消息给目标对象
#import "ViewController.h"
#import "objc/runtime.h"@interface MYPerson: NSObject@end@implementation MYPerson- (void)foo {NSLog(@"Doing foo");//MYPerson的foo函数
}@end@interface ViewController ()@end@implementation ViewController- (void)viewDidLoad {[super viewDidLoad];// Do any additional setup after loading the view, typically from a nib.//执行foo函数[self performSelector:@selector(foo)];
}// resolveInstanceMethod方法,用于动态解析未实现的方法
// 返回YES表示尝试继续转发过程
+ (BOOL)resolveInstanceMethod:(SEL)sel {return YES;//返回YES,进入下一步转发
}// forwardingTargetForSelector方法,寻找可以响应此选择器的其他对象
// 返回nil表示无合适对象,继续向后转发
- (id)forwardingTargetForSelector:(SEL)aSelector {return nil;//返回nil,进入下一步转发
}// methodSignatureForSelector方法,为尚未识别的选择器提供方法签名
// 当选择器为"foo"时,提供一个适合的签名,以便能够调用forwardInvocation
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {if ([NSStringFromSelector(aSelector) isEqualToString:@"foo"]) {return [NSMethodSignature signatureWithObjCTypes:"v@:"];//签名,进入forwardInvocation} return [super methodSignatureForSelector:aSelector];
}// forwardInvocation方法,用于处理无法直接识别的消息转发
// 这里创建一个MYPerson实例,并尝试在此实例上调用原始选择器
- (void)forwardInvocation:(NSInvocation *)anInvocation {SEL sel = anInvocation.selector;MYPerson *p = [MYPerson new]; // 创建MYPerson的实例if([p respondsToSelector:sel]) { // 检查MYPerson实例是否能响应此选择器[anInvocation invokeWithTarget:p]; // 能响应则在MYPerson实例上调用}else {[self doesNotRecognizeSelector:sel]; // 否则,报告选择器未识别}
}@end
结果不错
从打印结果来看,我们实现了完整的转发
通过签名,Runtime生成了一个对象anInvocation,发送给了forwardInvocation
我们在forwardInvocation方法里面让Person对象去执行了foo函数
签名参数v@:的解释在苹果文档Type Encodings有详细的解释
here
就这张图
参考博客
OC对象的本质(中)
iOS runtime详解
Objective-C Runtime