Objective-C是一门动态语言,它依赖runtime库将编译和链接时做的事情推迟到运行时处理。
例如我们在调用实例方法[receiver message]时,Objective-C会利用runtime转换成objc_msgSend(receiver,selector)。如果receiver所在的类方法列表中没有实现message的实现,则会去父类的方法列表中继续寻找...直到根类,如果一直没有响应message方法,那么就会进入消息动态处理机制。
Objective-C依次给我们三次拯救程序的机会:
1.动态添加方法
+(BOOL)resolveInstanceMethod:(SEL)sel { //动态添加实例方法
// 1.判断没实现方法,就动态添加方法
if(sel == @selector(method)){
//动态添加方法
class_addMethod(self,sel,(IMP)newRun,"v@:");
return YES;
}
return[super resolveInstanceMethod:sel];
}
+(BOOL)resolveClassMethod:(SEL)sel {//动态添加类方法
// 1.判断没实现方法,就动态添加方法
if(sel == @selector(method)){
/**动态添加方法。objc_getMetaClass("Class”)不能写成self,因为当我们对类发送消息时,会在该类的MetaClass的methodLists里查找。*/
class_addMethod(objc_getMetaClass("Class"),sel,(IMP)newRun,“v@:”);
return YES;
}
return[super resolveClassMethod:sel];
}
2.消息动态转发
-(id)forwardingTargetForSelector:(SEL)aSelector {
Object*object =[ObJectnew];
if([object respondsToSelector:aSelector]){ //如果Object类可以实现aSelector方法,那么转发到Object类去
return object;
}
return[super forwardingTargetForSelector:aSelector];
}
3.标准消息转发
//方法签名.
-(NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
//手动生成签名
if([NSStringFromSelector(aSelector);isEqualToString:@"method"]){
return[NSMethodSignature signatureWithObjCTypes:"v@:"];
} else {
return[super methodSignatureForSelector:aSelector];
}
}
//转发消息
-(void)forwardInvocation:(NSInvocation *)anInvocation {
// 1.拿到消息
SEL selector =[anInvocation selector];
// 2.转发消息.
Object*object =[ObJectnew];
if([object respondsToSelector:selector]){
//转发
[anInvocation invokeWithTarget:object];
} else {
[super forwardInvocation:anInvocation];
}
}
如果以上的方法都未能拯救你的程序,那么程序就会抛出异常,调用-(void)doesNotRecognizeSelector:(SEL)aSelector。
如果我们重载了该函数,就可以保证程序不会crash。但是苹果文档强调程序不能以此函数结束掉,必须抛出异常。