iOS开发过程中我们经常会碰到这样的报错:unrecognized selector sent to instance **,原因是我们调用了一个不存在的方法。用OC消息机制来说就是:消息的接收者不过到对应的selector,这样就启动了消息转发机制,我们可以通过代码在消息转发的过程中告诉对象应该如何处理未知的消息,默认实现是抛出异常
下面我们通过实例来看一下在抛出异常之前也就是消息转发过程中都经过了哪些步骤:
第一步:对象在收到无法解读的消息后,首先会调用实类方法+(BOOL)resolveInstanceMethod:(SEL)sel
或者类方法+ (BOOL)resolveClassMethod:(SEL)sel, 询问是否有动态添加方法来进行处理,处理实例如下
//动态解析方法
void sends(id self,SEL _cmd,NSString *message){
NSLog(@"%@",message);
}
+(BOOL)resolveInstanceMethod:(SEL)sel{
NSString *methodNme = NSStringFromSelector(sel);
// if ([methodNme isEqualToString:@"seenMessage:"]) {
// return class_addMethod(self,sel, (IMP)sends, "");
// }
return [super resolveInstanceMethod:sel];
}
当person 收到了未知 seenMessage选择子的消息的时候,如果是实例方法会首选调用上文的resolveInstanceMethod:方法,方法内通过判断选择子然后通过class_addMethod方法动态添加了一个sends的实现方法来解决掉这条未知的消息,此时消息转发过程提前结束。
但是当person 收到seenMessage 这条未知消息的时候,第一步返回的是No或者 [super resolveInstanceMethod:sel],也就是没有动态新增实现方法的时候就会调用第二步
第二步:既然第一步已经问过了,没有新增方法,那就问问有没有别人能够帮忙处理一下啊,调用的是- (id)forwardingTargetForSelector:(SEL)aSelector这个方法
上文我们说到person接收到了一条选择子为seenMessage的未知消息,我们可以看到控制台已经打印了resolveInstanceMethod: seenMessage,代表第一步已经问过了,那么第二步问一下是否有别的类能帮忙处理吗?代码如下:
//备用接收者
-(id)forwardingTargetForSelector:(SEL)aSelector{
NSString *methodName = NSStringFromSelector(aSelector);
if ([methodName isEqualToString:@"seenMessage:"]) {
return [animal new];
} else {
return [super forwardingTargetForSelector:aSelector];
}
}
通过- (id)forwardingTargetForSelector:(SEL)aSelector的处理,bird能够处理这条消息,所以这条消息被animal成功处理,消息转发流程提前结束。控制台打印
第三步:调用- (void)forwardInvocation:(NSInvocation *)anInvocation,在调用forwardInvocation:之前会调用- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector方法来获取这个选择子的方法签名,然后在-(void)forwardInvocation:(NSInvocation *)anInvocation方法中你就可以通过anInvocation拿到相应信息做处理,实例代码如下
当person 收到一条 选择子为code 的消息的时候,前两步发现都没办法处理掉,走到第三步:
//签名
-(NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{
NSString *methodName = NSStringFromSelector(aSelector);
if ([methodName isEqualToString:@"seenMessage:"]) {
return [NSMethodSignature signatureWithObjCTypes:"v@:@"];
}else {
return [super methodSignatureForSelector:aSelector];
}
}
- (void)forwardInvocation:(NSInvocation *)anInvocation {
NSLog(@"forwardInvocation: %@", NSStringFromSelector([anInvocation selector]));
if ([anInvocation selector] == @selector(code)) {
[anInvocation invokeWithTarget:[animal new]];
}
}
那么最后消息未能处理的时候,还会调用到
- (void)doesNotRecognizeSelector:(SEL)aSelector这个方法,我们也可以在这个方法中做些文章,避免掉crash,但是只建议在线上环境的时候做处理,实际开发过程中还要把异常抛出来
-(void)doesNotRecognizeSelector:(SEL)aSelector{
NSLog(@"aasdasdasd");
}
附带完整代码:
person.h
@interface person : NSObject
-(void)seenMessage:(NSString *)msg;
@end
person.m
#import "person.h"
#import <objc/runtime.h>
#import "animal.h"
@implementation person
//动态解析方法
void sends(id self,SEL _cmd,NSString *message){
NSLog(@"%@",message);
}
+(BOOL)resolveInstanceMethod:(SEL)sel{
NSString *methodNme = NSStringFromSelector(sel);
// if ([methodNme isEqualToString:@"seenMessage:"]) {
// return class_addMethod(self,sel, (IMP)sends, "");
// }
return [super resolveInstanceMethod:sel];
}
//备用接收者
-(id)forwardingTargetForSelector:(SEL)aSelector{
NSString *methodName = NSStringFromSelector(aSelector);
if ([methodName isEqualToString:@"seenMessage:"]) {
return [animal new];
} else {
return [super forwardingTargetForSelector:aSelector];
}
}
//签名
-(NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{
NSString *methodName = NSStringFromSelector(aSelector);
if ([methodName isEqualToString:@"seenMessage:"]) {
return [NSMethodSignature signatureWithObjCTypes:"v@:@"];
}else {
return [super methodSignatureForSelector:aSelector];
}
}
//消息转发
-(void)forwardInvocation:(NSInvocation *)anInvocation{
SEL sel = [anInvocation selector];
animal *imal= [animal new];
if ([imal respondsToSelector:sel]) {
[anInvocation invokeWithTarget:imal];
} else {
[super forwardInvocation:anInvocation];
}
}
-(void)doesNotRecognizeSelector:(SEL)aSelector{
NSLog(@"aasdasdasd");
}
animal.m
#import "animal.h"
@implementation animal
-(void)seenMessage:(NSString *)message{
NSLog(@"%@-----",message);
}
@end