[iOS]从拾遗到Runtime(上)

2024-05-09 04:12
文章标签 ios runtime 拾遗

本文主要是介绍[iOS]从拾遗到Runtime(上),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

[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对象在内存中存放的信息包括

  1. isa指针
  2. 其他成员变量
/// 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

这篇关于[iOS]从拾遗到Runtime(上)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/972354

相关文章

安卓链接正常显示,ios#符被转义%23导致链接访问404

原因分析: url中含有特殊字符 中文未编码 都有可能导致URL转换失败,所以需要对url编码处理  如下: guard let allowUrl = webUrl.addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed) else {return} 后面发现当url中有#号时,会被误伤转义为%23,导致链接无法访问

【iOS】MVC模式

MVC模式 MVC模式MVC模式demo MVC模式 MVC模式全称为model(模型)view(视图)controller(控制器),他分为三个不同的层分别负责不同的职责。 View:该层用于存放视图,该层中我们可以对页面及控件进行布局。Model:模型一般都拥有很好的可复用性,在该层中,我们可以统一管理一些数据。Controlller:该层充当一个CPU的功能,即该应用程序

Golang进程权限调度包runtime

关于 runtime 包几个方法: Gosched:让当前线程让出 cpu 以让其它线程运行,它不会挂起当前线程,因此当前线程未来会继续执行GOMAXPROCS:设置最大的可同时使用的 CPU 核数Goexit:退出当前 goroutine(但是defer语句会照常执行)NumGoroutine:返回正在执行和排队的任务总数GOOS:目标操作系统NumCPU:返回当前系统的 CPU 核数量 p

iOS剪贴板同步到Windows剪贴板(无需安装软件的方案)

摘要 剪贴板同步能够提高很多的效率,免去复制、发送、复制、粘贴的步骤,只需要在手机上复制,就可以直接在电脑上 ctrl+v 粘贴,这方面在 Apple 设备中是做的非常好的,Apple 设备之间的剪贴板同步功能(Universal Clipboard)确实非常方便,它可以在 iPhone、iPad 和 Mac 之间无缝传输剪贴板内容,从而大大提高工作效率。 但是,iPhone 如何和 Wind

iOS项目发布提交出现invalid code signing entitlements错误。

1、进入开发者账号,选择App IDs,找到自己项目对应的AppId,点击进去编辑, 2、看下错误提示出现  --Specifically, value "CVYZ6723728.*" for key "com.apple.developer.ubiquity-container-identifiers" in XX is not supported.-- 这样的错误提示 将ubiquity

我的第一次份实习工作-iOS实习生-第三个月

第三个月 这个月有一个考核项目,是一个电子书阅读器,组长说很重要,是我的实习考核项目。 我的项目XTReader,这是我参考网上的一些代码,和模仿咪咕阅读做的,功能还不完善,数据的部分是用聚合数据做的。要收费的。   还有阅读页面,基本功能实现了一下。使用了autolayout,自适应布局,也是第一次用网络,第一次用数据库,第一次用自动布局。还有很多不足。 做了一周多,有个问题一直没

我的第一次份实习工作-iOS实习生-公司使用过的软件

bittorrentsync 素材,文件同步软件 cornerstone svn 软件开发合作 mark man 测量坐标的软件 SQLLite Manager 数据库操作软件

我的第一次份实习工作-iOS实习生-第二个月

第二个月 来公司过了一个月了。每天早上9点上班,到晚上6.30下班,上下班要指纹打卡,第一个月忘了打卡好多次(),然后还要去补打卡单。公司这边还安排了,工资卡办理,招商银行卡。开了一次新员工大会,认识了公司的一些过往,公司的要求等,还加了一下公司的企业QQ,还有其他的羽毛球群,篮球群。我加了下羽毛球群,也去打了一两次。第二个月的感受,感觉跟组里面的交流跟沟通都好少,基本上还有好多人不认识。想想也

我的第一次份实习工作-iOS实习生-第一个月

实习时间:2015-08-20 到 2015-12-25  实习公司;福建天棣互联有限公司 实习岗位:iOS开发实习生 第一个月: 第一天来公司,前台报道后,人资带我去我工作的地方。到了那,就由一个组长带我,当时还没有我的办公桌,组长在第三排给我找了一个位置,擦了下桌子,把旁边的准备的电脑帮我装了下,因为学的是iOS,实习生就只能用黑苹果了,这是我实习用的电脑。 帮我装了一下电脑后,开机

iOS如何隐藏系统状态栏

这里主要说明一下iOS7系统给状态栏的适配及隐藏带来的改变。 变化一: 不隐藏状态栏的情况下,StatusBar会直接显示在当前页面上,当前页面的会延伸到 StatusBar下方,顶到最上头。 这种显示方式在iOS7上是无法改变的,也无法通过设置或者配置类达到iOS6的状态栏效果。       所以在iOS7上进行页面布局的时候要考虑