iOS必备技能之Runtime(一)

Runtime 是一个比较底层的C语言的API,可以翻译为“运行时”。作为使用运行时机制的OC语言的底层,它在程序运行时把OC语言转换成了runtime的C语言代码。学习并理解runtime是OC学习历程中的不可或缺的一大块儿。

一、消息机制

调用方法的本质就是发送消息。

发送消息常见的有四个方法:

  • objc_msgSend 向一个类的实例发送消息,返回id类型数据。(这也是最常用的一个发送消息的方法)
  • objc_msgSend_stret 向一个类的实例发送消息,返回结构体类型数据。
  • objc_msgSendSuper 向一个类的实例的父类发送消息,返回id类型数据。
  • objc_msgSendSuper_stret 向一个类的实例的父类发送消息,返回结构体类型的数据。

在OC语言中,方法的真正实现是在程序运行的时候绑定的,假如一个方法只有声明,没有实现,调用后在编译阶段是不会出错的,真正报错是在运行的时候。

[receiver message]

以上方法在运行时会被转化为

//receiver是方法的调用者,selector是方法名
objc_msgSend(receiver, selector)
//如果有参数
objc_msgSend(receiver, selector, arg1, arg2, ...)

发送消息的原理

objc_msgSend为了完成动态绑定,进行了以下三步:

  1. 首先它要先根据方法名找到方法的具体实现程序,因为多态性,同一个方法在不同的类里面可以有不同的实现,所以查找主要依靠寻找receiver所在的类。
  2. 传递参数,调用该方法的实现程序。
  3. 把该程序的返回值作为方法自己的返回值。
//runtime中对类的定义
struct objc_class {
    Class isa  OBJC_ISA_AVAILABILITY;
#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;
    struct objc_method_list **methodLists                    OBJC2_UNAVAILABLE;
    struct objc_cache *cache                                 OBJC2_UNAVAILABLE;
    struct objc_protocol_list *protocols                     OBJC2_UNAVAILABLE;
#endif
} OBJC2_UNAVAILABLE;

//runtime中对实例的定义
struct objc_object {
    Class isa  OBJC_ISA_AVAILABILITY;
};

如上runtime中对类的定义,每一个类都有指向父类的指针(super_class)和一个方法调度表(objc_method_list **methodLists:根据方法名SEL查找该方法的具体实现的地址IMP),当向一个对象发送消息的时候,该对象通过isa指针找到该对象的类(实际上,实例的定义里面也只有这个指针,没有别的了),在类的调度表查找该方法名,当找不到的时候,通过指向父类的指针找到该类的父类,然后在该类的父类中继续查找该方法名,这样递归查找一直到NSObject类为止(NSProxy类除外,它不属于NSObject子类)。如果查找到该方法名,根据调度表找到该方法的实现的地址进行调用。如下图所示

Messaging Framework

为了加速发送消息的进程,runtime系统会把使用过的方法名和对应的内存地址缓存起来,每个类都有一个单独的缓存空间,其中包含自己类的方法和继承自父类的方法。在查找调度表之前,runtime系统会首先在缓存中进行查找。

使用隐藏的参数

当objc_msgSend找到方法的实现程序时,它调用这个程序并传递所有方法的参数给它,这其中还包含两个隐藏的参数:

  • 消息的接收对象
  • 调用方法的方法名(selector)

这两个参数虽然没有在方法中进行定义,但是你可以很方便地使用它们。消息的接收对象通过self来引用,方法名通过_cmd来引用。

- strange
{
    id  target = getTheReceiver();
    SEL method = getTheMethod();
 
    if ( target == self || method == _cmd )
        return nil;
    return [target performSelector:method];
}

获取方法的地址

避免动态绑定的唯一方法就是直接获得方法的地址然后把它当做函数一样来调用。当一个方法被连续多次执行,而你又不想每次都用消息机制造成额外的开支,这种办法就是一个合适的使用时机。
下面的例子展示了如何节省开支多次调用setFilled:方法

void (*setter)(id, SEL, BOOL);
int i;
 
setter = (void (*)(id, SEL, BOOL))[target
    methodForSelector:@selector(setFilled:)];
for ( i = 0 ; i < 1000 ; i++ )
    setter(targetList[i], @selector(setFilled:), YES);

通过methodForSelector:方法,你可以请求得到指向实现该方法的程序的指针,然后通过这个指针调用该程序。值的注意的是,参数和返回值要正确声明,而且参数中id和SEL要进行显式声明。


二、动态方法

假如你想动态地为方法提供实现,OC使用@dynamic实现了这个特性。

@dynamic propertyName;

这样就会通知编译器和这个属性相关的方法将会动态提供。你可以通过方法resolveInstanceMethod:resolveClassMethod:分别为类方法和实例方法动态地提供实现。

一个OC的方法其实就是由C语言的函数再加上至少两个参数(self和_cmd)组成的。

你可以把一个函数通过class_addMethod作为方法添加到一个类中去。给定以下一个函数:

void dynamicMethodIMP(id self, SEL _cmd) {
    // implementation ....
}

你可以通过resolveInstanceMethod:这个方法把上面的函数以方法名(resolveThisMethodDynamically)动态地添加到一个类(MyClass)里面。具体实现方式如下:

@implementation MyClass
+ (BOOL)resolveInstanceMethod:(SEL)aSEL
{
    if (aSEL == @selector(resolveThisMethodDynamically)) {
          class_addMethod([self class], aSEL, (IMP) dynamicMethodIMP, "v@:");
          return YES;
    }
    return [super resolveInstanceMethod:aSEL];
}
@end

这其中,class_addMethod这个方法有四个参数,第一个是要添加方法的类,第二个是要添加的方法名,第三个是这个方法的实现函数的指针(值的注意的是,这个函数必须显式地把self_cmd这两个参数写出来),第四个是方法的参数数组,在这里它是用的类型编码的方式进行表示的,因为方法一定含有self_cmd这两个参数,所以字符数组的第二个和第三个字符一定是"@:",第一个字符代表返回值,这里为空用“v”来表示。相关知识点请见下文。


三、类型编码

为了使runtime系统更加简洁,编译器把每个方法的返回值和参数的类型都分别使用一个字符来编码,然后再把它们关联到方法选择器(selector)上。因为这种编码方案在其它环境中也很实用,所以我们可以很方便地使用@encode()编译器指令来自定义类似的编码。

char *buf1 = @encode(int **);
char *buf2 = @encode(struct key);
char *buf3 = @encode(Rectangle);

一般来说,不管是基本类型,还是指针,或者结构体,或者联合体,甚至可以是类名,只要这个类型能够作为C语言中sizeof()的参数,那么它就能被进行编码。

下表便是已经定义了的类型编码,使用@encode()编译器指令自定义编码的时候一定要避开这些字符。


Objective-C type encodings

特别注意,OC不支持long double类型,因此@encode(long double)会返回字符“d",意义为double
结构体的类型编码是按照结构体内部的类型的顺序来表示的,比如

typedef struct example {
    id   anObject;
    char *aString;
    int  anInt;
} Example;

会被编码为:

{example=@*i}

由第一章内容可以得知,类的实例的定义是一个只包含isa指针的结构体,所以[NSObject class]会被编码为

{ NSObject=# }

具体应用方面,上一章class_addMethod最后一个参数就是使用的类型编码来表示的函数返回值和参数的类型。


参考:《Objective-C Runtime Programing Guide》


链接:
iOS必备技能之Runtime(二)

文章会不定期进行增添和更新,欢迎订阅和收藏!

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

推荐阅读更多精彩内容