data:image/s3,"s3://crabby-images/3acaa/3acaac9f9afa26970f3444e1ae44c2436a08f133" alt=""
1.数据结构
(1)objc_object
data:image/s3,"s3://crabby-images/be3d1/be3d160d3d4bcbf3024f00ccbef64d707621cb52" alt=""
(2)objc_class
data:image/s3,"s3://crabby-images/4313d/4313d8622c0c989a082004c641f254cf41d132d3" alt=""
(3)isa指针
data:image/s3,"s3://crabby-images/62b27/62b27b5245b7a5b3b89fc2f80ad70b2de0427221" alt=""
什么是isa指针?
isa分为指针型isa,isa的值代表class地址,非指针型isa的值的部分代表class的地址。
(4)isa指向
- 关于
对象
,其指向类对象
截图4.png
- 关于
类对象
,其指向元类对象
截图5.png
(5) catch_t
- 用于快速查找方法执行的函数。
- 可
可增量扩展
的哈希表
结构 - 是
局部性原理的
最佳应用
其数据结构
如下图
截图6.png
(6) class_data_bits_t
-
class_data_bits_t
主要是对class_rw_t
的封装 -
class_rw_t
代表了类相关的读写
信息,对class_ro_t的封装
(7) class_rw_t
data:image/s3,"s3://crabby-images/cf67a/cf67aa06bf88d3377a7f4ae22134214f50fd4e4d" alt=""
(8) class_ro_t
data:image/s3,"s3://crabby-images/09d32/09d321708942bbdc8b5a093cdf184e73e1ab9ecc" alt=""
(9) method_t
data:image/s3,"s3://crabby-images/b8033/b8033465d22a118533c7c36fc7d3ad5cc2ae28eb" alt=""
Type Encodings
- const char * types
截图10.png
整体数据结构
图11.png
2.对象、类对象、元类对象
-
类对象
存储实例方法列表等信息 -
元类对象
存储类方法列表等信息
截图12.png
类对象和元类对象之间有什么区别和联系?
实例对象可以通过isa指针
找到类对象,从而找到类对象中存储的实例方法列表等信息,类对 象可以通过isa指针
找到元类对象,从而找到元类对象中存储的类方法列表等信息,类对象和元类对象都是objc_class
这种数据结构的。对于任何一个元类对象,它的isa指针都指向根元类对象。根元类对象的superClass指针指向根类对象。
思考
调用一个类方法,会沿着元类对象及其元类对象的父类的方法列表依次查找,当类方法在元类对象中找不到时,就会找根类对象中,同名的实例方法实现。
消息传递过程
调用实例方法, 首先系统会根据当前实例的isa指针,找到对应的类对象,在类对象及其父类的方法列表中依次查找,直到找到根类对象,如果还是没用找到,就会走到消息的转发流程。
调用类方法,系统会根据当前类的isa指针,找到对应的元类对象,在元类对象及其元类对象的父类方法列表中依次查找,直到找到根元类,如果根元类中也没有,就会到根类对象中查找同名的实例方法,如果最终还是没找到就会走到消息的转发流程。
3.消息传递
data:image/s3,"s3://crabby-images/c9de2/c9de2ed94640ee765b97a4faa033aa96d7699a69" alt=""
data:image/s3,"s3://crabby-images/68c66/68c66d28df9b8472a1163f499b12963c18de1782" alt=""
data:image/s3,"s3://crabby-images/7b1c0/7b1c02c7a915cca27e0723e82d5dac0c5301066f" alt=""
#import "Phone.h"
@implementation Phone
- (instancetype)init{
self = [super init];
if (self) {
NSLog(@"%@",NSStringFromClass([self class]));
NSLog(@"%@",NSStringFromClass([super class]));
}
return self;
}
@end
最终的打印结构是一致的,都是Phone
(1)缓存查找
给定值SEL
,目标值是对应bucket_t
中的IMP
data:image/s3,"s3://crabby-images/c4432/c4432d2fa8e8670c2072eb5cad5a9e3262dd9f27" alt=""
首先根据方法选择器,通过一个函数映射出bucket_t在数组中的位置,这一步骤涉及到哈希查找。哈希查找就是根据给定的一个值,然后经过哈希函数的算法算出的值,就是给定的值的在数组中的索引位置。
(2)当前类中查找
- 对于
已排序好
的列表,采用二分查找
算法查找方法对应执行函数。 - 对于
没有排序
的列表,采用一般遍历
查找方法对应执行函数。
(3)父类逐级查找
data:image/s3,"s3://crabby-images/49d9c/49d9ccaf67316e2b94e31ded176e231c4a272e22" alt=""
4.消息转发
data:image/s3,"s3://crabby-images/855f2/855f293b38208c2ba5500dc2171a0f3003317fb5" alt=""
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
// Override point for customization after application launch.
RuntimeObject *obj = [[RuntimeObject alloc] init];
[obj test];
return YES;
}
#import <Foundation/Foundation.h>
@interface RuntimeObject : NSObject
- (void)test;
@end
#import "RuntimeObject.h"
@implementation RuntimeObject
+ (BOOL)resolveInstanceMethod:(SEL)sel{
//如果test方法 打印日志
if (sel == @selector(test)) {
NSLog(@"resolveInstanceMethod:");
return NO;
}else{
return [super resolveInstanceMethod:sel];
}
}
- (id)forwardingTargetForSelector:(SEL)aSelector{
NSLog(@"forwardingTargetForSelector");
return nil;
}
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{
if (aSelector == @selector(test)) {
NSLog(@"methodSignatureForSelector");
// v代表返回值是void类型 @代表第一个参数类型是id,即self
// : 代表第二个参数是SEL类型的,即@selector(test)
return [NSMethodSignature signatureWithObjCTypes:"v@:"];
}else{
return [super methodSignatureForSelector:aSelector];
}
}
- (void)forwardInvocation:(NSInvocation *)anInvocation{
NSLog(@"forwardInvocation");
}
打印结果:
data:image/s3,"s3://crabby-images/36032/36032660b66ad4e680f02c1cea185e4a1025a522" alt=""
5.Method-Swizzling
data:image/s3,"s3://crabby-images/d85be/d85becc0fd573aec09666501e8dd4db36f3c43d5" alt=""
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
RuntimeObject *obj = [[RuntimeObject alloc] init];
[obj test];
return YES;
}
#import <Foundation/Foundation.h>
@interface RuntimeObject : NSObject
- (void)test;
- (void)otherTest;
@end
#import "RuntimeObject.h"
#import <objc/runtime.h>
@implementation RuntimeObject
+ (void)load{
//获取test方法
Method test = class_getInstanceMethod(self, @selector(test));
//获取otherTest
Method otherTest = class_getInstanceMethod(self, @selector(otherTest));
//交换两个方法的实现
method_exchangeImplementations(test, otherTest);
}
- (void)test{
NSLog(@"test");
}
- (void)otherTest{
//实际上调用test的具体实现
[self otherTest];
NSLog(@"otherTest");
}
+ (BOOL)resolveInstanceMethod:(SEL)sel{
//如果test方法 打印日志
if (sel == @selector(test)) {
NSLog(@"resolveInstanceMethod:");
return NO;
}else{
return [super resolveInstanceMethod:sel];
}
}
- (id)forwardingTargetForSelector:(SEL)aSelector{
NSLog(@"forwardingTargetForSelector");
return nil;
}
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{
if (aSelector == @selector(test)) {
NSLog(@"methodSignatureForSelector");
// v代表返回值是void类型 @代表第一个参数类型是id,即self
// : 代表第二个参数是SEL类型的,即@selector(test)
return [NSMethodSignature signatureWithObjCTypes:"v@:"];
}else{
return [super methodSignatureForSelector:aSelector];
}
}
- (void)forwardInvocation:(NSInvocation *)anInvocation{
NSLog(@"forwardInvocation");
}
@end
打印结果
data:image/s3,"s3://crabby-images/fc9e6/fc9e607acc951ce547bcffc54383c71c9dd5bdf7" alt=""
6.动态添加方法
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
RuntimeObject *obj = [[RuntimeObject alloc] init];
[obj test];
return YES;
}
#import <Foundation/Foundation.h>
@interface RuntimeObject : NSObject
- (void)test;
@end
#import "RuntimeObject.h"
#import <objc/runtime.h>
@implementation RuntimeObject
void testImp(void)
{
NSLog(@"test invoke");
}
+ (BOOL)resolveInstanceMethod:(SEL)sel{
//如果test方法 打印日志
if (sel == @selector(test)) {
NSLog(@"resolveInstanceMethod:");
class_addMethod(self, @selector(test),testImp,"v@:");
return YES;
}else{
return [super resolveInstanceMethod:sel];
}
}
打印结果
data:image/s3,"s3://crabby-images/9967b/9967b80102e082a6917e825663d3a58b5fbd7989" alt=""
网友评论