ios开发专题

ios 消息转发

2018-06-07  本文已影响3人  zl520k

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));

   //增加提示

}

以上是消息转发的整个流程。   

上一篇下一篇

猜你喜欢

热点阅读