事岀无常必有妖-iOS捉妖记之(Runtime)

Runtime是开源的,任何时候你都可以从http://opensource.apple.com获取。事实上查看 Objective-C 源码是我理解它是如何工作的第一种方式,在某些问题上要比读苹果的文档要好。

引言

相信很多从事iOS开发的小伙伴们都听过这样一句形容runtime的话:

runtime就像是iOS开发中的妖怪,谁都听说过,但少有人见(用)到过!

这句话是某知名培训机构内某老师对学生们说的一句话,相信不少人尤其是初学的萌新们还没了解过runtime,听了这句话就被吓到了!直接在心里给runtime打个一打标签[危险,慎用,底层,难,用不到,不用掌握]。以至于很多人做了有一段时间的iOS开发却依然对其一知半解……

定义

Objective-C 的 Runtime 是一个运行时库(Runtime Library),它是一个主要使用 C 和汇编写的库,为 C 添加了面相对象的能力并创造了 Objective-C。这就是说它在类信息(Class information) 中被加载,完成所有的方法分发,方法转发,等等。Objective-C runtime 创建了所有需要的结构体。

其实就在下个人的理解:runtime就是丫Objective-C 的灵魂!Objective-C之所以叫Objective-C是因为他比C语言不同,是面向对象的。但是Objective-C为什么有面相对象的能力?就是因为有runtime这个鬼东西!

进阶

我们为什么要学习runtime?

  • runtime可以遍历对象的属性
  • runtime可以动态添加/修改属性,动态添加/修改/替换方法,动态添加/修改/替换协议
  • runtime可以动态创建类/对象/协议等等
  • runtime可以方法拦截调用

其实runtime所能做的还不止这些,你甚至可以利用它来把一个Class A的实例对象a在程序中当作Class B的实例对象来用。所以很多iOS开发者把runtime叫做obj-C的黑魔法!

常用方法

先来个最简单最基本的也是几乎所有runtime文必备的例子:

obj-C: [obj func];

runtime:objc_msgSend(obj, @selector(func);

很多初学者除了知道runtime把对象的方法调用转化成消息发送的代码之后就不知道其他的了,但是显然仅仅知道上述的转化并没有什么“吡-”用,我们来看runtime中比较常用(实用)的几种基本用法:

  • 遍历对象的属性

首先定义一个简单的类Person

@interface Person : NSObject

@property (nonatomic, copy) NSString *name;

@property (nonatomic, assign) NSInteger age;

@end

然后在需要遍历对象的属性时

id personClass = objc_getClass(“Person”);

unsigned int outCount;

objc_property_t *properties = class_copyPropertyList(personClass, &outCount);

for (int i = 0; i < outCount; ++i) {

objc_property_t property = properties[i];

printf(“%s:%s\n”, property_getName(property), property_getAttributes(property));

}

free(properties);

这时就会打印出这个类对象的属性相关信息:

name:T@”NSString”,C,N,V_name

age:Tq,N,V_age

  • 消息转发

[消息转发]指的就是我上面提到的动态方法解析,重定向以及消息转发,我们先来看一张图:

动态方法解析:

从上图可以知道,当对一个实例对象obj发送一条消息func时[obj func],当前obj如果没有对func实现对应的方法,那么就runtime会调用+ (BOOL)resolveInstanceMethod:(SEL)sel方法允许开发者对当前受到的消息func做出响应,这就是动态方法解析。

继续拿上面的Person举例子,给Person类加一个体重weight属性

@property (nonatomic, assign) NSInteger weight;

然后在.m文件中加入一下代码

@implementation Person

@dynamic weight;  //避免自动生成getter/setter方法

//重写resolveInstanceMethod方法,动态方法解析

+ (BOOL)resolveInstanceMethod:(SEL)sel

{

if (sel == @selector(setWeight:)) {

class_addMethod([self class], sel, (IMP)setPropertyDynamic, “v@:”);

return YES;

}

return [super resolveInstanceMethod:sel];

}

//用来响应setWeight的c语言方法

void setPropertyDynamic(id self, SEL _cmd) {

NSLog(@”Dynamic setWeight”);

}

@end

然后可以在代码就调用Person的setWeight方法

Person *lision = [[Person alloc] init];

lision.weight = 75;

这时候如果不重写+ (BOOL)resolveInstanceMethod:(SEL)sel方法本应该异常的,但是你可以发现程序会打印出信息:

Dynamic setWeight

重定向:

那么还是看图说话,如果没有重写+ (BOOL)resolveInstanceMethod:(SEL)sel方法,那就就会调用- (id)forwardingTargetForSelector:(SEL)aSelector方法,把这个消息让另一个对象来处理,这次叫做重定向。

跟着上面的例子走,先另一个类People用来等待重定向:

@interface People : NSObject

@end

给新写的People类加一个weight方法,但是注意:People没有weight属性!

– (NSInteger)weight

{

return 70;

}

接下来我们重写- (id)forwardingTargetForSelector:(SEL)aSelector方法:

– (id)forwardingTargetForSelector:(SEL)aSelector

{

if (aSelector == @selector(weight)) {

People *people = [[People alloc] init];

return people;

}

return [super forwardingTargetForSelector:aSelector];

}

然后我们在刚才的执行代码中:

NSLog(@”weight = %ld”, lision.weight);

然后运行,经历过上面的例子你肯定知道不会异常啦,而且你会发现虽然你给weight属性赋值明明是75,可是打印结果是:weight = 70。这就是Person类- (id)forwardingTargetForSelector:(SEL)aSelector方法中把这条信息抛给了people对象,调用了People类的weight方法!

消息转发:

那么如果上面的两个方法都没有重写,并且消息依然是当前对象没有实现的方法,runtime才会启用消息转发调用– (void)forwardInvocation:(NSInvocation *)anInvocation,需要注意的是很多文章没有提到这个方法花费代价较大,如果要实现把消息转发类似的功能建议最好使用重定向,而且再调用这个方法前runtime会先调用- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector方法。

我们跟着上面的例子,继续给Person类加入属性:

@property (nonatomic, copy) NSString *ID;

以及上面提到的两个方法:

– (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector

{

if (aSelector == @selector(setID:)) {

NSMethodSignature *sig = [NSMethodSignature signatureWithObjCTypes:”v@:”];

//”v@:”代表的意思参见Objective-C Type Encodings,这里的意思是返回值为空

return sig;

}

return nil;

}

– (void)forwardInvocation:(NSInvocation *)anInvocation

{

People *people = [[People alloc] init];

if ([people respondsToSelector:anInvocation.selector]) {

[anInvocation invokeWithTarget:people];

}

}

别忘记了在People类中添加对应的方法:

– (void)setID:(NSString *)ID

{

NSLog(@”People setID: %@”, ID);

}

最后,我门只需要在执行代码块中加入代码:

lision.ID = @”xxxx”;

结果显而易见,相信各位都知道将会打印信息:

People setID: xxxx

写在最后

其实runtime就是我们无时无刻不在用的东西,只是人们习惯对看不到的东西怀有恐惧心理而已。我们平时的obj-C代码都是被runtime转译为c和汇编语言运行的。我个人认为大公司为什么喜欢在面试时问runtime相关的东西是因为大公司往往不仅仅要会干活的人,它还会要求这些会干活的人知道其中的原理!我们自己也应该要求自己或多或少的理解这些原理,知道我们为什么写出的obj-C代码经历了哪些过程run到我们的设备上,不要敲了很多年的代码还是一只只会干活的码农。

发表评论