关于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 <Foundation/Foundation.h>

@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中的消息转发。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,269评论 19 139
  • 通过前边的学习我们知道,某个类或者对象调某个方法实际上就是给这个类/对象发送消息,如果我们某个对象要调用某个方法,...
    啊啊啊啊锋阅读 250评论 0 0
  • 前言 本篇文章是研究消息转发的机制,苹果的消息转发机制就像一条链,消息传送链越长则消耗也越大,最好是在第一级就可以...
    G_GUI阅读 222评论 0 0
  • 最近受到哲学“海”老师的启发,开始引导孩子也自己阅读些非“向善”类的书籍。于是,读了这本东野圭吾的《恶意》。 作家...
    雨霏Maggie妈阅读 236评论 1 0
  • ###四个收获 1、知道了分类是任何管理的基础,知道了为什么公司、职场要分等级了,因为是为了便于管理,一个人也就只...
    成长范阅读 198评论 0 0