iOS 源码解析 - Runtime篇(1)

objc-runtime 开源地址

在Objective-C中runtime主要充当了一个消息传递者,但其实它还有很多更加强大的特性。

工程打开之后可以看到如下结构

  • Public Headers 部分 这是我们平常能够用到的api包括
    • NSObjCRuntime.h
    • NSObject .h NSObject类的本体,Foundation里也有Foundation/NSObject.h 但里面只是一些分类和protocol
    • message.h
    • objc-api.h
    • objc-auto.h
    • objc-sync.h
    • objc-exception.h
    • objc.h
    • runtime.h

我们主要从上述几个头文件来展开源码阅读。

1.我们先从最熟悉的NSObject.h 类开始。与runtime相关的首先是两个属性:

Class isa  OBJC_ISA_AVAILABILITY; 
Class superclass;

我们用一张图说明它们无与伦比的重要性!


20170307194653265.png

由于oc设计模式也是基于原型模式,所以在oc中一切皆对象的说法也适用。

可以看到NSObject的继承体系离不开这两条线,isa指针负责 "横向指向",superclass指针负责 "纵向指向"。runtime的方法调用大部分也是依托于它们来获取方法列表。具体会在runtime的消息机制部分涉及。

这里有人可能会问,Instance of Subclass的superclass指针指向哪里呢?
我们可以在源码中找到答案:
- (Class)superclass {
    return [self class]->superclass;
}
- (Class)class {
    return object_getClass(self);
}
Class object_getClass(id obj){
    if (obj) return obj->getIsa();
    else return Nil;
}
可以看到其实实例对象通过isa指着获取当前类对象,然后直接拿到类对象的superclass 所以他们是共用的一个superclass

其实Instance of Subclass属于另外一种类型:

struct objc_object {
private:
    isa_t isa;
}
  1. 除了上边两个属性,NSObject.h中还有几个方法和runtime关系密切!
  • 我们先说定义在名为NSObject protocol中的方法:
- (id)performSelector:(SEL)aSelector;
- (id)performSelector:(SEL)aSelector withObject:(id)object;
- (id)performSelector:(SEL)aSelector withObject:(id)object1 withObject:(id)object2;

- (BOOL)respondsToSelector:(SEL)aSelector;

前三者的实现大同小异

- (id)performSelector:(SEL)sel {
    if (!sel) [self doesNotRecognizeSelector:sel];
    return ((id(*)(id, SEL))objc_msgSend)(self, sel);
}

- (id)performSelector:(SEL)sel withObject:(id)obj {
    if (!sel) [self doesNotRecognizeSelector:sel];
    return ((id(*)(id, SEL, id))objc_msgSend)(self, sel, obj);
}

- (id)performSelector:(SEL)sel withObject:(id)obj1 withObject:(id)obj2 {
    if (!sel) [self doesNotRecognizeSelector:sel];
    return ((id(*)(id, SEL, id, id))objc_msgSend)(self, sel, obj1, obj2);
}

先判断SEL类型的参数是否存在,SEL类型是一个方法签名。我们可以用这两个方法来获取SEL:

 NSSelectorFromString(@"selectorName");
 @selector(selectorName);

关于SEL的生成方法,可能由于不同框架下的生成方法不一致,苹果并没有给出具体的实现,但是在objc-sel.mm文件中我们可以看到一个类似实现,这也说明了SEL的本质其实就是单纯对方法名字的一个定向处理:

static SEL sel_alloc(const char *name, bool copy)
{
    selLock.assertWriting();
    return (SEL)(copy ? strdup(name) : name);    
}

通过这个SEL当做key来寻找具体的函数实现地址(即IMP);

值得一提的是每次创建SEL前都会从namedSelectors这个NXMapTable hash表中去查找缓存下来的SEL,如果这里面没有找到才会去在内存中创建,并且插入到缓存表中。

我们再回退到刚才的三个方法中。
当SEL不存在的时候,会触发

 - (void)doesNotRecognizeSelector:(SEL)sel {
    _objc_fatal("-[%s %s]: unrecognized selector sent to instance %p", 
                object_getClassName(self), sel_getName(sel), self);
}

最终打印log日志,生成crash日志,调用__builtin_trap函数触发内核陷阱,交还控制权,安全退出程序。

_objc_syslog(buf2); 
_objc_crashlog(buf2);
_objc_trap();

接下来如果SEL存在,则调用objc_msgSend()函数,runtime中最重要的函数!
这个函数真正开启了runtime的消息机制!苹果源码里给出了它的汇编代码的实现。篇幅有限,下篇见

NSObject.mm 中还有很多内容,比如Autorelease pool的实现,准备在Runloop篇在进行阅读。
另外还有一些针对于Swift的处理,笔者会选择性的忽略它,放到以后Swift篇中说明。

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

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,678评论 0 9
  • 参考链接: http://www.cnblogs.com/ioshe/p/5489086.html 简介 Runt...
    乐乐的简书阅读 2,129评论 0 9
  • 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的转载 这篇文章完全是基于南峰子老师博客的...
    西木阅读 30,540评论 33 466
  • 本文转载自:http://yulingtianxia.com/blog/2014/11/05/objective-...
    ant_flex阅读 744评论 0 1
  • 今天是 30天习惯养成计划 的三十一天,我做了下面这些事情。 [x] 6:50起床 [x] 起床后喝一杯温开水 [...
    _尔东陈_阅读 206评论 0 0