ios 消息转发
ios在类中,没有定义的函数,要走消息转发流程。如果不走消息转发流程,程序会奔溃。消息转发流程分四步调用。
第一步:+ (BOOL)resolveInstanceMethod:(SEL)sel,这个方法会调用,如果在这个方法中使用
runtime中的class_addMethod这个方法,可以将没有定义的方法,进行动态增加。
例如:下面的person类
@interface Person : NSObject
- (void)createPerson:(NSString*)str;
@end
@implementation Person
+ (BOOL)resolveInstanceMethod:(SEL)sel{
NSLog(@"%@",NSStringFromSelector(sel));
if(sel ==@selector(createPerson:)) {
class_addMethod(self, sel,(IMP) newInterRun, "v@:@:");
return YES;
}
return [super resolveInstanceMethod:sel];
}
void newInterRun(id self,SEL sel,NSString str) {
NSLog(@"---newInterRun---%@",str);
}
如果在resolveInstanceMethod方法中,增加了方法,消息转发流程就结束了。
如果在resolveInstanceMethod方法中,没有增加方法,直接[super resolveInstanceMethod:sel];,就转发到第二步。
第二步:- (id)forwardingTargetForSelector:(SEL)aSelector,这个方法是消息转发其他类中定义当前aSelector的方法,
例如:其他类SubPerson
@interfaceSubPerson :NSObject
- (void)createPerson:(NSString*)str;
@end
@implementation SubPerson
- (void)createPerson:(NSString*)str{
NSLog(@"---createPerson---%@",str);
}
@end
- (id)forwardingTargetForSelector:(SEL)aSelector{
NSLog(@"%@",NSStringFromSelector(aSelector));
return [SubPerson new];
}
如果在forwardingTargetForSelector方法中明确转发特定的类时,此时消息转发结束;
如果在forwardingTargetForSelector方法中没有明确转特定的类是,此时要继续转发和重新签名,然后进入第三步
- (id)forwardingTargetForSelector:(SEL)aSelector{
NSLog(@"%@",NSStringFromSelector(aSelector));
return [super forwardingTargetForSelector:aSelector];
}
- (NSMethodSignature*)methodSignatureForSelector:(SEL)aSelector{
if(aSelector ==@selector(createPerson:)) {
NSMethodSignature *signature = [NSMethodSignature signatureWithObjCTypes:"v@:@:"];
returnsignature;
}
else{
return [super methodSignatureForSelector:aSelector];
}
}
如果不签名,直接调用[super methodSignatureForSelector:aSelector];,导致整个流程会崩溃的。
第三步:- (void)forwardInvocation:(NSInvocation*)anInvocation,这个方法是最后特定类的方法进行调用
- (void)forwardInvocation:(NSInvocation*)anInvocation{
NSLog(@"%@",anInvocation);
SEL sel = anInvocation.selector;
SubPerson *p = [SubPerson new];
if ([p respondsToSelector:sel]) {
[anInvocation invokeWithTarget:p];
}
else{
[superforwardInvocation:anInvocation];
}
}
如果在SubPerson类中定义sel,转发流程结束。
如果在SubPerson类中没有定义sel,转发到第四步。
第四步:- (void)doesNotRecognizeSelector:(SEL)aSelector,调用这个方法,进行提示,当前aSelector,在整个消息转发流程中,都没有定义这个,然后进行提示。
- (void)doesNotRecognizeSelector:(SEL)aSelector{
NSLog(@"%@",NSStringFromSelector(aSelector));
//增加提示
}
以上是消息转发的整个流程。