OC方法的本质探索

前言

前面一篇文章我们知道了缓存的插入 cache_t::insert 最后找到了 objc_msgSend 这个方法,那么 objc_msgSend 到底是什么这篇文章就来了解一下。

比如一个你调用一个 [person say666]; 的方法,底层到底是怎么样的一套流程呢?

运行时

在此之前,我们先补充一下几个概念,运行时以及编译时。

  • 编译时
    顾名思义就是正在编译的时候,那啥叫编译呢?就是编译器帮你把源代码翻译成机器能识别的代码。
    那编译时就是简单的作一些翻译工作,比如检查你有没有粗心写错啥关键字了啊,有啥词法分析,语法分析之类的过程。就像个老师检查学生的作文中有没有错别字和病句一样,如果发现啥错误编译器就告诉你让你直接去修改。所谓这时的错误就叫编译时错误,这个过程中做的啥类型检查也就叫编译时类型检查,或静态类型检查(所谓静态就是没把真把代码放内存中运行起来, 而只是把代码当作文本来扫描下)所以有时一些人说编译时还分配内存啥的肯定是错误的说法。

  • 运行时
    运行时(runtime)就是代码跑起来了,被装载到内存中去了,而运行时类型检查就与前面讲的编译时类型检查(或者静态类型检查)不一样,不是简单的扫描代码而是在内存中做些操作,做些判。运行时就是尽可能多的将编译期的事情放到运行期去决议。

运行时发起的三种方式

  • OC 层直接调用
    OC层的直接调用酒喝我们日常开发中的方法调用时一幕一样的,比如 CDPerson *person = [[CDPerson alloc] init];[person say666];这些方法就是OC层的调用。
  • NSObject 层的提供的方法
    NSObject 层的就是和OC层差不多一样,只是NSObject 封装好了的,比如[person isKindOfClass:[CDPerson class]];,[person performSelector:@selector(say666)];这些方法。
  • C/C++ 提供的API
    C/C++提供的API就是底层的api,OC源码经过编译器编译就会变成C/C++ 的代码,比如 Class cls = objc_getClass("CDPerson");,IMP imp = class_getMethodImplementation(cls, @selector(say2));这些就是C/C++层的。
三种方式

那么我们如何体现OC层的源代码到底层就是C/C++了呢?这里我们借助Clang。

objc_msgSend 的初探

准备一份简单的代码如下

@interface CDPerson : NSObject
- (void)say666;
+ (void)sayHello;
- (void)say1;
- (void)say2;
- (void)say3;
- (void)say4;
- (void)saySomthing:(NSString *)sth with:(NSString *)text;
@end

@implementation CDPerson
- (void)say666 { NSLog(@"say666"); }
+ (void)sayHello { NSLog(@"sayHello"); }
- (void)say1 { NSLog(@"%s", __func__); }
- (void)say2 { NSLog(@"%s", __func__); } 
- (void)saySomthing:(NSString *)sth with:(NSString *)text { NSLog(@"%@ - %@", sth, text); }
@end

///如下一个调用一下这个方法。
[CDPerson sayHello];
CDPerson *person = [CDPerson alloc];
[person say666]; 
NSLog(@"%@", person);

然后我们通过clang 把这份源码编译成c++的源码看看到底是什么样呢。通过源码我们可以看到OC 层面的代码在底层是怎样的了

/// 等价于:[CDPerson sayHello];
((void (*)(id, SEL))(void *)objc_msgSend)((id)objc_getClass("CDPerson"), sel_registerName("sayHello"));
/// 等价于:CDPerson *person = [CDPerson alloc];
CDPerson *person = ((CDPerson *(*)(id, SEL))(void *)objc_msgSend)((id)objc_getClass("CDPerson"), sel_registerName("alloc"));
/// 等价于:[person say666]; 
        ((void (*)(id, SEL))(void *)objc_msgSend)((id)person, sel_registerName("say666"));
/// NSLog(@"%@", person);
        NSLog((NSString *)&__NSConstantStringImpl__var_folders_xt_dwtgnm2n0dg4fcph5yx7p0k00000gn_T_main_7865ca_mi_0, person);

从上面的源代码可以知道,OC层的方法(函数)在编译成底层c++ 后都变成 objc_msgSend 了。

既然这样了,我们OC层是对C/C++的封装,那我们是否可以直接调用底层的 objc_msgSend 来调用方法呢?于是乎我们按照C++ 的格式来写一个方法

((void (*)(id, SEL))(void *)objc_msgSend)((id)person, @selector(say1));

结果是编译通过,并且运行也正确。那么我们看看 objc_msgSend 到底是怎样实现的?于是乎我们打开 objc 的源码进入 objc_msgSend 里面,可以看到如下的定义,但是找了一圈都没有发现具体的实现,最后发现在汇编语言里面有调用相关的实现。可以知道 objc_msgSend 是用汇编实现的(这个以后在讨论)。

OBJC_EXPORT void
objc_msgSend(void /* id self, SEL op, ... */ )
    OBJC_AVAILABLE(10.0, 2.0, 9.0, 1.0, 2.0);

OBJC_EXPORT void
objc_msgSendSuper(void /* struct objc_super *super, SEL op, ... */ )
    OBJC_AVAILABLE(10.0, 2.0, 9.0, 1.0, 2.0);

到这里我在想是否可以更加简单的去调用 objc_msgSend 呢?于是乎这样调用

objc_msgSend(person, @selector(say2));

但是结果总确实不可以的,出现了如下的错误。


直接调用 objc_msgSend

出现如下的错误是因为上面的 objc_msgSend 定义的地方有个宏定义 #if !OBJC_OLD_DISPATCH_PROTOTYPES。而在以前的xCode上这样调用时没有问题的。

#if defined(__arm64__) && !__swift__
#   undef OBJC_OLD_DISPATCH_PROTOTYPES
#   define OBJC_OLD_DISPATCH_PROTOTYPES 0
#endif

解决办法是 Xcode -> Targets -> Build Setting -> Enable Strict Checking of objc_msgSend Calls -> NO

最后我们在来调用一个 [person say3]看看结果又是如何,结果依然可以编译通过。但是当我们运行的时候却抱错了?相信大家对于这个错误都很熟悉


这个也说明了我们的编译器在编译期只是单单的做一些错误排查。由于我们的objc_msgSend 是运行时调用的,所以这样的错误在编译期是体现不出来的,因为我们这种错误可以在运行时处理。

** 总结 **

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,271评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,275评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,151评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,550评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,553评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,559评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,924评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,580评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,826评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,578评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,661评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,363评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,940评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,926评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,156评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,872评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,391评论 2 342

推荐阅读更多精彩内容