不得不说的runtime

不得不说的runtime

前言


最近工作上事情比较琐碎,但是还是抽出点时间跟着冰霜大神的思路进精神病院深入学习了runtime的源码并成功出院,因此,想写一篇文章记录下自己的收获。

在这里我并不想写得很仔细,因为网上资料很多,只想轻轻点一下,介绍个大概。

实例、类和元类


刚开始接触runtime的时候,相信很多人认为runtime就是一个工具库,可以给classificatory添加属性,可以Method Swizzling,但是并不理解是它是怎么实现的,只是照葫芦画瓢,会用而已。其实作为一个IOS开发者,这是远远不够的,想要了解runtime的魅力,还得从实例、类和元类说起。

class

NSObject的定义如下

typedef struct objc_class *Class;

@interface NSObject <NSObject> {
    Class isa  OBJC_ISA_AVAILABILITY;
}

我们可以看到,NSObject中就只有一个isa,而这个isa其实就是一个objc_class结构体。

在Objc2.0之前,objc_class源码如下:

struct objc_class {
    Class isa  ;

#if !__OBJC2__
    Class super_class                                               OBJC2_UNAVAILABLE;// 父类指针
    const char *name                                         OBJC2_UNAVAILABLE; //类名
    long version                                             OBJC2_UNAVAILABLE; 
    long info                                                OBJC2_UNAVAILABLE;
    long instance_size                                       OBJC2_UNAVAILABLE; //类大小
    struct objc_ivar_list *ivars                             OBJC2_UNAVAILABLE; //(*ivars)成员变量指针列表
    struct objc_method_list **methodLists                    OBJC2_UNAVAILABLE; //(*methodLists)方法列表指针,列表里面记录的是该类的方法
    struct objc_cache *cache                                 OBJC2_UNAVAILABLE; //方法缓存列表记录,里面记录的是最近使用过的方法
    struct objc_protocol_list *protocols                     OBJC2_UNAVAILABLE; //所有协议列表
#endif

} OBJC2_UNAVAILABLE;

而在Objc2.0之后,objc_class被改造成了这个样子:

typedef struct objc_class *Class;
typedef struct objc_object *id;

@interface Object { 
    Class isa; 
}

@interface NSObject <NSObject> {
    Class isa  OBJC_ISA_AVAILABILITY;
}

struct objc_object {
private:
    isa_t isa;
}

struct objc_class : objc_object {
    // Class ISA;
    Class superclass;
    cache_t cache;             // formerly cache    pointer and vtable
    class_data_bits_t bits;    // class_rw_t * plus custom rr/alloc flags
}

union isa_t 
{
    isa_t() { }
    isa_t(uintptr_t value) : bits(value) { }
    Class cls;
    uintptr_t bits;
}

虽然样子变化很大但是基本结构还是和Objc2.0之前一致,因此我们还是对着Objc2.0之前的样子来解释。关于为什么Category只能添加方法,不能添加私有变量的原因,我见过这么一个解释:
Category工作在运行时,objc_class是一个结构体,这个时候objc_class结构体的大小是确定了,如果给ivars添加一个私有变量,那么整个objc_class的大小就改变了(多了私有变量的大小),这就不行了。而methodLists只是一个指针,添加一个方法就相当于换一个数组更大的指针给objc_class,不同指针的大小是一样的,因此并没有改变objc_class的大小,因此可以实现。

ps:这是网上某篇文章的解释,对此还是有点疑惑。

isa和superclass

关于superclass,学过面向对象语言的同学应该都很清楚是什么意思,比如BaseViewController和SpecialViewController,SpecialViewController继承自BaseViewController,那么SpecialViewController的superClass就是BaseViewController。

其实isa的关系和这个有点像:

@interface Son : NSObject
@end
  int main(int argc, char * argv[]) {
  Son * boy = [[Son alloc]init];
  }

在这个例子中 boy的isa指针就指向Son这个类。换句话说,一个类实例化出来的对象的isa指向这个类

下面要介绍一个概念,元类(meta class)。其实每个类中也有一个isa指针,也就是说,类也是其他类实例化出来的对象!而这个"其他类"指的就是元类。元类实例化出类,类实例化出对象,这大概就是元类的定位。那么元类有什么用,在下面的消息发送机制中会有介绍。

最后再来看看一张经典的图:


isa和superclass

其中要解释的是,Root class就可以理解成NSObject,虚线是isa,实线是superclass,所有的元类的isa都指向NSObject,于是整个体系就完成了。

消息发送 & 消息转发


消息转发机制

这张图很好得介绍了整个消息发送以及转发的过程。

objc_msgSend函数简介

我们来看看最简单的oc函数调用:

[self call];

一般的函数调用都满足这样的形式[receiver message],在编译时会被编译器转化为:

id objc_msgSend ( id target, SEL op, ... );

这里objc_msgSend会做这么几件事:
1、检测这个selector是不是要忽略的,如果忽略不操作。
2、检查target是不是nil,如果没有处理nil的函数,就自动清理现场并返回。(防止给nil发消息崩溃)
3、确定不是给nil发消息之后,根据该对象的isa找到该类,再从该类的缓存方法列表里找方法对应的IMP实现。
4、如果缓存方法列表找不到,则在方法列表中找。
5、如果方法列表中也没找到,则根据super指针找到父类的class,继续执行4操作,最后到NSObject中。
6、如果NSObject中也没该方法则执行消息转发,该过程后面讲。
7、转发也没人接就崩溃。

消息发送阶段图

这里我想插一下讲一讲有关元类的东西。

@interface Son : NSObject
- (void)call;
+ (void)call2;
@end

 int main(int argc, char * argv[]) {
  Son * boy = [[Son alloc]init];
  [boy call];
  [Son call2];
  }

[boy call][Son call2]有什么区别呢?

根据上面的流程, [boy call] 会先找到boy的isa指向的对象,也就是Son这个然后在Son中查找call方法,也就是说,- (void)call;存储在Son的方法列表里,也就是说,类存储"-"的方法(实例方法)。

[Son call2]会先找Son的isa指向的对象,也就是Son的元类查找call2方法,然后执行。也就是说,元类存储"+"的方法(类方法)。

这也就能解释为什么[boy call2]编译器会报错,因为boy的isa是Son,Son中找不到call2方法。并且也能说明元类很重要,因为它存储着该类的类方法。

消息转发

关于消息转发很多初学者应该并不知道有这么一层,而且这层非常有意思。AOP(面向切面编程)就是根据这个原理来实现。

上面消息发送如果找不到方法,那么就会在该对象的类中执行- (id)forwardingTargetForSelector:(SEL)aSelector去查找备用的接收对象。如果该对象能respondsToSelector该方法,则交给该备用对象处理该方法。否则进行下一步。

- (NSMethodSignature *)methodSignatureForSelector:(SEL)selector- (void)forwardInvocation:(NSInvocation *)anInvocation是配合执行,methodSignatureForSelector抛出doesNotRecognizeSelector异常,forwardInvocation接收并处理。

下面是一个简单的demo,自己下个断点尝试就能了解其流程:

#import <UIKit/UIKit.h>
#import <objc/runtime.h>
#import "Son.h"

@interface Student : NSObject
@end

@implementation Student

- (NSMethodSignature *)methodSignatureForSelector:(SEL)selector
{
    NSString *sel = NSStringFromSelector(selector);

        return [NSMethodSignature signatureWithObjCTypes:"@@:"];

}
- (void)forwardInvocation:(NSInvocation *)anInvocation
{
    if ([Son instancesRespondToSelector:[anInvocation selector]]) {
        [anInvocation invokeWithTarget:[[Son alloc]init]];
    }
    else {
        [super forwardInvocation:anInvocation];
    }

}
- (id)forwardingTargetForSelector:(SEL)aSelector    {
    //如果注释掉就执行methodSignatureForSelector
    if(aSelector == @selector(call)) {
        return [[Son alloc]init];
    }
    return [super forwardingTargetForSelector:aSelector];
}

@end

@interface Son : NSObject
@end

@implementation Son

- (void)call{
    NSLog(@"%@", NSStringFromClass([self class]));
}
@end

int main(int argc, char * argv[]) {
    Student * st  = [[Student alloc]init];
    [st performSelector:@selector(call) withObject:nil];
}

运用这两个步骤可以对一个类的方法进行"偷梁换柱"从而在不破坏源码的情况下插入代码片段。

最后


想记的也就这么多了,后面关于Method Swizzling等都是怎么使用runtime,相信大家或多或少都有用过,因此不再继续叙述。

喜欢oc的理由不仅仅是因为其优雅简洁的语言,更应该是其runtime特性给开发者带来无限的想象空间。然而到了swift时代,runtime就变得非常麻烦,这点还是非常令人惋惜的。

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

推荐阅读更多精彩内容

  • 转至元数据结尾创建: 董潇伟,最新修改于: 十二月 23, 2016 转至元数据起始第一章:isa和Class一....
    40c0490e5268阅读 1,678评论 0 9
  • 我们常常会听说 Objective-C 是一门动态语言,那么这个「动态」表现在哪呢?我想最主要的表现就是 Obje...
    Ethan_Struggle阅读 2,170评论 0 7
  • 转载:http://yulingtianxia.com/blog/2014/11/05/objective-c-r...
    F麦子阅读 727评论 0 2
  • 巜杨树》/前行者 仲夏,家乡的杨树林美极了,枝茂叶繁,树木青葱,杨树,以它独特的气质和魅力,叹为观止。 色彩,...
    前行者1一常德一自由人阅读 905评论 0 0
  • 有时候觉得自己活的太不潇洒,有太多的束缚太多的压抑使我们不敢做自己,时间长了也就真的变的俗气,离心目中的形象越来越...
    发现精彩阅读 233评论 0 0