消息转发

消息转发_第1张图片
消息处理流程

方案一:
判断是否能够相应事件

+ (BOOL)resolveInstanceMethod:(SEL)sel
+ (BOOL)resolveClassMethod:(SEL)sel

方案二:
当不能响应时,返回一个接受消息的 taget

- (id)forwardingTargetForSelector:(SEL)aSelector

方案三:
返回签名,如果methodSignatureForSelector返回 nil 即签名失败,forwardInvocation将造成崩溃.

- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector;
- (void)forwardInvocation:(NSInvocation *)anInvocation;

首先系统会调用resolveInstanceMethod (如果是类方法则调用resolveClassMethod) 可以在这个方法里动态的添加绑定方法.

返回 YES 则代表可以实现方法
可以用NSString *selStr = NSStringFromSelector(sel);来获取方法名

如果现在自定义一个Person类 让它实现 run 方法,但是在.m中不实现 run 方法
代码如下

#import "Person.h"
#import 
@implementation Person
+(BOOL)resolveInstanceMethod:(SEL)sel{
    NSString *selName = NSStringFromSelector(sel);
        if ([selName isEqualToString:@"run"]) {
            class_addMethod(self, sel, (IMP)run, "v@:");
        }
    return [super resolveClassMethod:sel];
}
void run(id self,SEL _cmd){
    NSLog(@"%@ -- %s",self,sel_getName(_cmd));
}
1.person 会先实现resolveInstanceMethod 方法,判断本类能否实现方法,我在resolveInstanceMethod 方法中给类绑定了 run 方法所以 super resolveInstanceMethod 将会返回 YES 而 run 方法也得以运行

有一点我比较奇怪:为什么,在新绑定了 run 方法后为什么resolveInstanceMethod无论返回 YES 或者 NO 都可以执行 run 方法

2.当resolveInstanceMethod没有绑定 run 方法, 时会执行forwardingTargetForSelector方法 ,这个方法返回你需要转发消息的对象。
3.如果我们不实现forwardingTargetForSelector,系统就会调用方案三的两个方法methodSignatureForSelector和forwardInvocation

methodSignatureForSelector用来生成方法签名,这个签名就是给forwardInvocation中的参数NSInvocation调用的。

错误unrecognized selector sent to instance原因,原来就是因为methodSignatureForSelector这个方法中,由于没有找到run对应的实现方法,所以返回了一个空的方法签名,最终导致程序报错崩溃。

TIP:self代表方法调用者,_cmd代表这个方法的SEL,签名类型就是用来描述这个方法的返回值、参数的,v代表返回值为void,@表示self,:表示_cmd。

参(摘)考(抄):
http://www.cocoachina.com/ios/20150604/12013.html
http://www.jianshu.com/p/01a19c64499c

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