毕竟汇编语言代码比较晦涩难懂,因此这里将函数的实现反汇编成C语言的伪代码:

//下面的结构体中只列出objc_msgSend函数内部访问用到的那些数据结构和成员。

/*

其实SEL类型就是一个字符串指针类型,所描述的就是方法字符串指针

*/

typedef char * SEL;

/*

IMP类型就是所有OC方法的函数原型类型。

*/

typedef id (*IMP)(id self, SEL _cmd, ...);

/*

objc_msgSend的C语言版本伪代码实现.

receiver: 是调用方法的对象

op: 是要调用的方法名称字符串

*/

id  objc_msgSend(id receiver, SEL op, ...)

{

//1............................ 对象空值判断。

//如果传入的对象是nil则直接返回nil

if (receiver == nil)

return nil;

//2............................ 获取或者构造对象的isa数据。

void *isa = NULL;

//如果对象的地址最高位为0则表明是普通的OC对象,否则就是Tagged Pointer类型的对象

if ((receiver & 0x8000000000000000) == 0) {

struct objc_object  *ocobj = (struct objc_object*) receiver;

isa = ocobj->isa;

}

else { //Tagged Pointer类型的对象中没有直接保存isa数据,所以需要特殊处理来查找对应的isa数据。

//如果对象地址的最高4位为0xF, 那么表示是一个用户自定义扩展的Tagged Pointer类型对象

if (((NSUInteger) receiver) >= 0xf000000000000000) {

//自定义扩展的Tagged Pointer类型对象中的52-59位保存的是一个全局扩展Tagged Pointer类数组的索引值。

int  classidx = (receiver & 0xFF0000000000000) >> 52

isa =  objc_debug_taggedpointer_ext_classes[classidx];

}

else {

//系统自带的Tagged Pointer类型对象中的60-63位保存的是一个全局Tagged Pointer类数组的索引值。

int classidx = ((NSUInteger) receiver) >> 60;

isa  =  objc_debug_taggedpointer_classes[classidx];

}

}

//因为内存地址对齐的原因和虚拟内存空间的约束原因,

//以及isa定义的原因需要将isa与上0xffffffff8才能得到对象所属的Class对象。

struct objc_class  *cls = (struct objc_class *)(isa & 0xffffffff8);

//3............................ 遍历缓存哈希桶并查找缓存中的方法实现。

IMP  imp = NULL;

//cmd与cache中的mask进行与计算得到哈希桶中的索引,来查找方法是否已经放入缓存cache哈希桶中。

int index =  cls->cache.mask & op;

while (true) {

//如果缓存哈希桶中命中了对应的方法实现,则保存到imp中并退出循环。

if (cls->cache.buckets[index].key == op) {

imp = cls->cache.buckets[index].imp;

break;

}

//方法实现并没有被缓存,并且对应的桶的数据是空的就退出循环

if (cls->cache.buckets[index].key == NULL) {

break;

}

//如果哈希桶中对应的项已经被占用但是又不是要执行的方法,则通过开地址法来继续寻找缓存该方法的桶。

if (index == 0) {

index = cls->cache.mask;  //从尾部寻找

}

else {

index--;   //索引减1继续寻找。

}

} /*end while*/

//4............................ 执行方法实现或方法未命中缓存处理函数

if (imp != NULL)

return imp(receiver, op,  ...); //这里的... 是指传递给objc_msgSend的OC方法中的参数。

else

return objc_msgSend_uncached(receiver, op, cls, ...);

}

/*

方法未命中缓存处理函数:objc_msgSend_uncached的C语言版本伪代码实现,这个函数也是用汇编语言编写。

*/

id objc_msgSend_uncached(id receiver, SEL op, struct objc_class *cls)

{

//这个函数很简单就是直接调用了_class_lookupMethodAndLoadCache3 来查找方法并缓存到struct objc_class中的cache中,最后再返回IMP类型。

IMP  imp =   _class_lookupMethodAndLoadCache3(receiver, op, cls);

return imp(receiver, op, ....);

}

https://www.jianshu.com/p/df6629ec9a25

objc_msgSend函数的实现的更多相关文章

  1. 初探 objc_msgSend函数

    1.0 执行某个对象的方法    [receiver message] 被编译为: id objc_msgSend(id self,SEL op,...): objc_msgSend 发送信息的过程 ...

  2. 继承自NSObject的不常用又很有用的函数(2)

    函数调用 Objective-C是一门动态语言,一个函数是由一个selector(SEL),和一个implement(IML)组成的.Selector相当于门牌号,而Implement才是真正的住户( ...

  3. objc_msgSend()报错Too many arguments to function call ,expected 0,have3

    Build Setting--> Apple LLVM 6.0 - Preprocessing--> Enable Strict Checking of objc_msgSend Call ...

  4. objc_msgSend消息传递学习笔记 – 消息转发

    该文是 objc_msgSend消息传递学习笔记 – 对象方法消息传递流程 的基础上继续探究源码,请先阅读上文. 消息转发机制(message forwarding) Objective-C 在调用对 ...

  5. objc_msgSend消息传递学习笔记 – 对象方法消息传递流程

    在Effective Objective-C 2.0 – 52 Specific Ways to Improve Your iOS and OS X Programs一书中,tip 11主要讲述了Ob ...

  6. 为什么objc_msgSend必须用汇编实现

    译者前言 总是看到有人说用汇编实现objc_msgSend是为了速度快,当然这个不可否认.但是难道没有别的原因?于是就看到了这篇文章,遂翻译之!=.= 我自己的理解就是,用汇编实现,是为了应对不同的“ ...

  7. OC语言的特性(一)-消息传递与调用函数的表现形式

    我们在初学Objective-C时,都会觉得ObjC中的消息传递和其他语言的调用函数差不多,只是在OC中,方法调用用消息传递这一概念来代替. 那么到底怎样区别OC中的消息传递与其他语言的调用函数呢. ...

  8. runtime objc_msgSend

    runtime objc_msgSend 字数1781 阅读245 评论2 喜欢7  前言 想要通过runtime发送消息,就必须要掌握runtime如何发送消息,是调用哪个函数?又是如何调用的?本篇 ...

  9. AFNetworking 3.0 源码解读(五)之 AFURLSessionManager

    本篇是AFNetworking 3.0 源码解读的第五篇了. AFNetworking 3.0 源码解读(一)之 AFNetworkReachabilityManager AFNetworking 3 ...

随机推荐

  1. 啰哩吧嗦式讲解在windows 家庭版安装docker

    1.docker是什么,为什么要使用docker Docker 是一个开源的应用容器引擎,让开发者可以打包他们的应用以及依赖包到一个可移植的容器中, 然后发布到任何流行的 Linux 机器上,也可以实 ...

  2. 第二节:Java开发环境的搭建

    一.认识并安装JDK 1.JDK(Java Development Kit)是Java开发工具集,包括Java运行环境(JRE).Java开发工具以及一些基础类库,进行Java开发所必须安装的软件. ...

  3. 网络编程: 基于UDP协议的socket

    udp是无链接的,启动服务之后可以直接接受消息,不需要提前建立链接 UDP协议的通信优势: 允许一个服务器同时和多个客户端通信, TCP不行 服务端 import socket sk = socket ...

  4. 在arcgis使用python脚本进行字段计算时对中文的处理方案

    一.引言 在arcgis打开一个图层的属性表,可以对属性表的某个字段进行计算,但是在平常一般都是使用arcgis提供的字段计算器的界面进行傻瓜式的简答的赋值操作,并没有使用到脚本对字段值进行逻辑的操作 ...

  5. loj#2542. 「PKUWC2018」随机游走(MinMax容斥 期望dp)

    题意 题目链接 Sol 考虑直接对询问的集合做MinMax容斥 设\(f[i][sta]\)表示从\(i\)到集合\(sta\)中任意一点的最小期望步数 按照树上高斯消元的套路,我们可以把转移写成\( ...

  6. BZOJ3238: [Ahoi2013]差异(后缀自动机)

    题意 题目链接 Sol 前面的可以直接算 然后原串翻转过来,这时候变成了求任意两个前缀的最长公共后缀,显然这个值应该是\(len[lca]\),求出\(siz\)乱搞一下 #include<bi ...

  7. 正能量:You Are the Best

    Success comes from knowing that you did your best to become the best that you are capable of becomin ...

  8. 64位版本的Windows不兼容,masm无法运行解决方法

    问题: 在Window64位运行不了的masm 解决方法: 1.下载DosBox0.74(当前最新): 2.安装后运行,运行后出现控制台: 3.在DosBox的控制台下运行 Mount x: x:/m ...

  9. RPC簡介

    RPC 技术原理       RPC ( Remote Procedure Call Protocol,远程过程调用协议 ): 客户端在不知道调用细节的情况下,调用存在于远程计算机上的某个对象,就像调 ...

  10. redis 持久化策略、aof配置、测试、手动持久化、aof文件体积优化

    redis持久化策略 1.数据文件.rdb 2.更新日志.aof 设置aof 1.命令方式config set appendonly noconfig rewrite2.配置文件方式 redis持久化 ...