深入Objective-C的动态特性
Objective-C具有相当多的动态特性,经常被提到和用到的有动态类型(Dynamic typing)、动态绑定(Dynamic binding)和动态加载(Dynamic loading)。
动态特性基础
1、动态类型
即运行时在决定对象的类型。例如:id类型,可以指向任意对象类型,在实际的开发中,会使用introspection来确定该对象实际所属类:
id obj = [Person new];
if ([obj isKindOfClass:[Person class]])
{
Person *person = (Person *)obj;
// Do something .....
}
-isKindOfClass:
是 NSObject 的方法,用以确定某个 NSObject 对象是否是某个类的成员。
-isKindOfClass:
是NSObject的方法,用以确定某个对象是否是某个类或其子类的成员。
这两个方法为典型的introspection方法。在确定对象为某类成员后,可以安全地进行强制转换,继续之后的工作。
2、动态绑定
基于动态类型,一个对象实例化之后,其类型就被确定了。该对象的属性和响应的函数也被确定下来,这就是动态绑定。
在继续之前,需要明确Objective-C中消息的概念。由于OC的动态特性,在OC中其实很少提及“函数”的概念,传统的函数一般在编译时就已经把参数信息和函数实现打包到编译后的源码中了,而在OC中最常使用的是消息机制。
调用一个实例的方法,所做的是向该实例的指针发送消息,实例在收到消息后,从自身的实现中寻找响应这条消息的方法。
动态绑定所做的就是实例所属类确定后,将某些属性或者方法动态绑定到实例上,当然这里的属性和方法是类中没有实现的,而是在运行时新加入的实现。
在Cocoa层,我们一般向一个NSObject对象发送-respondsToSelector:或者-instancesRespondToSelector:等来确定对象是否可以对某个SEL做出响应,而在OC消息转发机制被触发之前,对应的类的+resolveClassMethod:和+resolveInstanceMethod:将会被调用,在此时有机会动态地向类或者实例添加新的方法,也即类的实现是可以动态绑定的。一个例子:
//调用
id obj = [Person new];
if ([obj isKindOfClass:[Person class]])
{
Person *person = (Person *)obj;
//判断是否响应这个方法,如果找不到则会调用resolveInstanceMethod
if ([person respondsToSelector:@selector(printMethod)])
{
//因为是运行期间调用,所以不能直接调用对应的方法名
[person performSelector:@selector(printMethod)];
}
}
//实现
- (void)logMessage
{
NSLog(@"============================");
}
//类方法调用之前,调用此方法
+ (BOOL)resolveClassMethod:(SEL)sel
{
NSLog(@"resolveClassMethod");
return [super resolveClassMethod:sel];
}
//实例方法调用之前,调用此方法
+ (BOOL)resolveInstanceMethod:(SEL)sel
{
NSLog(@"resolveInstanceMethod");
//这里的printMethod和调用时的方法保持一致性原则。
if (sel == @selector(printMethod))
{
class_addMethod([self class], sel, class_getMethodImplementation_stret(self, @selector(logMessage)), "s@:@");
return YES;
}
return [super resolveInstanceMethod:sel];
}
当然实现动态绑定的需求方法有很多,这里简单举几个例子:
//给分类添加属性
objc_setAssociatedObject()
objc_getAssociatedObject()
//Method Swizzling(黑魔法,方法交换)
method_exchangeImplementations
3、动态加载
根据需求动态地加载资源。对于iOS来说,基本上就是根据不同的机型做适配。
例如:在4.7寸的屏幕上加载@2x的图片,在5.5寸的屏幕上加载@3x的图片。
有任何问题欢迎指出,我会及时纠正问题,谢谢!