关于iOS消息转发

今天去YY面试,问起了消息转发,竟然一时答不出来。
现在把iOS消息转发的流程过一遍。
首先我们要知道消息转发都有哪些方法以及其的调用流程,才能更好的掌握。

// 比如调用的方法没有实现或者不存在的时候OC会让你有机会选择哪个方法继续执行
//实例方法先调用:
+ (BOOL)resolveInstanceMethod:(SEL)sel
//类方法不存在时先调用
+ (BOOL)resolveClassMethod:(SEL)sel;

如果在以上的方法都没有做处理,OC还会给你第二个机会
-(id)forwardingTargetForSelector:(SEL)aSelector;

如果在第二次机会还没有处理,还有最后一次机会
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector;
- (void)forwardInvocation:(NSInvocation *)anInvocation;

举个栗子:
现在准备两个对象分别是DogPig

//Dog.h
#import 

@interface Dog : NSObject

-(void)eat;
+(void)sleep;

@end
#import "Dog.h"

@implementation Dog
@end

此处Dog.m是没有实现-eat+sleep方法的
调用eat方法时会报一下错误

2018-03-06 00:09:06.337035+0800 MassageForward[2059:18708264] -[Dog eat]: unrecognized selector sent to instance 0x100513f10
2018-03-06 00:09:06.338884+0800 MassageForward[2059:18708264] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[Dog eat]: unrecognized selector sent to instance 0x100513f10'

现在我们在Don.m加上最开始的那几个方法,变成了以下样子

void eatSomthing(id self, SEL sel)
{
    NSLog(@"%@ %s 【eat something】", self, sel_getName(sel));
}

// 决定选择哪个实例方法
+ (BOOL)resolveInstanceMethod:(SEL)sel
{
    NSLog(@"%s %@",__func__,NSStringFromSelector(sel));

//    动态添加一个方法,如果不添加跳到Step 2
//        if(sel == @selector(eat))
//        {
//            // 添加一个代替eat实现的方法
//            class_addMethod(self, sel, (IMP)eatSomthing, "v@:");
//            return YES;
//        }

    return [super resolveInstanceMethod:sel];
}



//当类方法不存在时调用 如: +eat()没有实现的话
+ (BOOL)resolveClassMethod:(SEL)sel
{
    NSLog(@"%s %s",__func__, sel_getName(sel));

    return [super resolveClassMethod:sel];
}


//=============================================Step 2=========================================================

-(id)forwardingTargetForSelector:(SEL)aSelector
{
    NSLog(@"%s %@",__func__,NSStringFromSelector(aSelector));
    
    //转发给Pig的实例,调用pig的eat方法,如果不转发则跳到Step 3
    //    return [Pig new];
    
    return [super forwardingTargetForSelector:aSelector];
}

//=============================================Step 3=========================================================
// 签名
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
{
    NSLog(@"%s %@",__func__,NSStringFromSelector(aSelector));
    
    if(aSelector == @selector(eat))
    {
        return [NSMethodSignature signatureWithObjCTypes:"v@:"];
    }
    
    return [super methodSignatureForSelector:aSelector];
}

- (void)forwardInvocation:(NSInvocation *)anInvocation
{
    NSLog(@"%s",__func__);

    SEL selector = [anInvocation selector];
    Pig *pig = [Pig new];

    if([pig respondsToSelector:selector])
    {
        [anInvocation invokeWithTarget:pig];
    }
}

这样子,就可以实现,在没有实现eat或者sleep方法时,实现了消息转发,从而不会导致程序崩溃。
要说明的是"v@:",每个方法会有两个默认值,一个是self_cmd, 表示方法的拥有者和SEL, 签名类型就是描述这个方法的参数和返回值的
其中v表示void, @表示self:表示_cmd

此示例中就是Dog调用eat方法后被Pigeat方法接收了。这就是OC中的消息转发。

你可能感兴趣的:(关于iOS消息转发)