最近在重温Effective Objective-C 2.0,这篇文章属于重温的产物吧,本文会通过demo来讲解OC中的消息转发机制

Demo:点我查看,觉得有帮助的话不要吝惜你的star
话不多说,iOS开发过程中我们经常会碰到这样的报错:unrecognized selector sent to instance **,原因是我们调用了一个不存在的方法。用OC消息机制来说就是:消息的接收者不过到对应的selector,这样就启动了消息转发机制,我们可以通过代码在消息转发的过程中告诉对象应该如何处理未知的消息,默认实现是抛出下面的异常

unrecognized selector.png
下面我们通过实例来看一下在抛出异常之前也就是消息转发过程中都经过了哪些步骤:


第一步:对象在收到无法解读的消息后,首先会调用+(BOOL)resolveInstanceMethod:(SEL)sel或者+ (BOOL)resolveClassMethod:(SEL)sel, 询问是否有动态添加方法来进行处理,处理实例如下

1
2
3
4
5
6
7
8
9
10
11
12
13
//People.m
void speak(id self, SEL _cmd){
NSLog(@"Now I can speak.");
}
+ (BOOL)resolveInstanceMethod:(SEL)sel {
NSLog(@"resolveInstanceMethod: %@", NSStringFromSelector(sel));
if (sel == @selector(speak)) {
class_addMethod([self class], sel, (IMP)speak, "V@:");
return YES;
}
return [super resolveInstanceMethod:sel];
}

当People 收到了未知 speak选择子的消息的时候,如果是实例方法会首选调用上文的resolveInstanceMethod:方法,方法内通过判断选择子然后通过class_addMethod方法动态添加了一个speak的实现方法来解决掉这条未知的消息,此时消息转发过程提前结束。
但是当People 收到fly 这条未知消息的时候,第一步返回的是No,也就是没有动态新增实现方法的时候就会调用第二步


第二步:既然第一步已经问过了,没有新增方法,那就问问有没有别人能够帮忙处理一下啊,调用的是- (id)forwardingTargetForSelector:(SEL)aSelector这个方法
上文我们说到People接收到了一条选择子为fly的未知消息,我们可以看到控制台已经打印了resolveInstanceMethod: fly,代表第一步已经问过了,那么第二步问一下是否有别的类能帮忙处理吗?代码如下:

1
2
3
4
5
6
7
8
9
10
11
12
- (id)forwardingTargetForSelector:(SEL)aSelector {
NSLog(@"forwardingTargetForSelector: %@", NSStringFromSelector(aSelector));
Bird *bird = [[Bird alloc] init];
if ([bird respondsToSelector: aSelector]) {
return bird;
}
return [super forwardingTargetForSelector: aSelector];
}
// Bird.m
- (void)fly {
NSLog(@"I am a bird, I can fly.");
}

通过- (id)forwardingTargetForSelector:(SEL)aSelector的处理,bird能够处理这条消息,所以这条消息被bird成功处理,消息转发流程提前结束。控制台打印

1
2
forwardingTargetForSelector: fly
I am a bird, I can fly.

但是如果- (id)forwardingTargetForSelector:(SEL)aSelector也找不到能够帮忙处理这条未知消息,那就会走到最后一步,这步也是代价最大的一步


第三步:调用- (void)forwardInvocation:(NSInvocation *)anInvocation,在调用forwardInvocation:之前会调用- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector方法来获取这个选择子的方法签名,然后在-(void)forwardInvocation:(NSInvocation *)anInvocation方法中你就可以通过anInvocation拿到相应信息做处理,实例代码如下

当People 收到一条 选择子为code 的消息的时候,前两步发现都没办法处理掉,走到第三步:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
- (void)forwardInvocation:(NSInvocation *)anInvocation {
NSLog(@"forwardInvocation: %@", NSStringFromSelector([anInvocation selector]));
if ([anInvocation selector] == @selector(code)) {
Monkey *monkey = [[Monkey alloc] init];
[anInvocation invokeWithTarget:monkey];
}
}
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
NSLog(@"method signature for selector: %@", NSStringFromSelector(aSelector));
if (aSelector == @selector(code)) {
return [NSMethodSignature signatureWithObjCTypes:"V@:@"];
}
return [super methodSignatureForSelector:aSelector];
}

这时控制台会打印

1
2
3
4
5
resolveInstanceMethod: code
forwardingTargetForSelector: code
method signature for selector: code
forwardInvocation: code
I am a coder.

此时这个code消息已经被monkey实例处理掉
此时消息转发流程完整的结束了,完整的消息转发流程如下:


那么最后消息未能处理的时候,还会调用到
- (void)doesNotRecognizeSelector:(SEL)aSelector这个方法,我们也可以在这个方法中做些文章,避免掉crash,但是只建议在线上环境的时候做处理,实际开发过程中还要把异常抛出来

EOF:OC中消息转发流程大概就是这样了,Demo点这里,觉得有帮助的话不要吝惜你的star,由于个人能力有限,文中难免有些错误,希望大家不吝赐教~
另外有一个问题想问大家,+ (BOOL)resolveClassMethod:(SEL)sel 在这个方法中怎么动态添加类方法? 比如我发送了一条未知的 [People missMethod]消息,怎么添加 +(void)missMethod 的实现呢?