iOS–消息转发机制
相信大家对这句话unrecognized selector sent to instance 0x*********
一点都不陌生吧。
下面就来简单说一下 拯救即将崩溃代码–iOS的消息转发
动态绑定引发
因为OC是一个动态运行时语言,其中之一的特性就是动态绑定。
关于动态绑定,苹果官网的给的解释为:(determining the method to invoke at runtime)。
简单点来说就是:程序直到执行时才能确定实际要调用的方法。
这样就会造成一个问题,我可以向一个实例发送一个消息,让它执行一个不属于自己的方法。这个时候就会出现unrecognized selector sent to instance。
如果发生这种情况,那么我们就可以应用消息转发来解决这个问题。把这个不属于自己的方法变成属于自己的方法,或者找一个有这个方法的实例来执行这个方法。
拯救即将崩溃代码
第一步 方法解析处理阶段 | 动态方法决议
该方法内可为当前类动态添加方法。
将sel的方法实现指向一个已存在的方法
/**
方法解析处理阶段 | 动态方法决议
该方法内可为当前类动态添加方法。
将sel的方法实现指向一个已存在的方法
*/
+ (BOOL)resolveInstanceMethod:(SEL)sel
{
NSString *selectorString = NSStringFromSelector(sel);
printf("%s %s\n", __func__, selectorString.UTF8String);
// 根据 sel 得到 class 的实例方法
Method method = class_getInstanceMethod([self class], @selector(dynamic_method));
// 根据 sel 得到 class 的函数指针
IMP method_imp = class_getMethodImplementation([self class], @selector(dynamic_method));
// 给找不到实现的sel添加实现
BOOL ret = class_addMethod([self class], sel, method_imp, method_getTypeEncoding(method));
printf("%s\n", ret?"交换添加成功":"交换添加失败");
// 返回结果不影响流程
return YES;
}
第二步 快速转发阶段 | 快速查找
上一步未解决问题时触发。
返回一个能响应aSelector的实例,即将aSelector转发给另外的类。
/**
快速转发阶段 | 快速查找
上一步未解决问题时触发。
返回一个能响应aSelector的实例,即将aSelector转发给另外的类。
*/
- (id)forwardingTargetForSelector:(SEL)aSelector
{
NSString *selectorString = NSStringFromSelector(aSelector);
printf("%s %s\n", __func__, selectorString.UTF8String);
if ([selectorString isEqualToString:@"no_imp_method"]) {
// 返回一个实现了aSelector函数的实例
// 如果该实例没有实现aSelector,则进入下一步methodSignatureForSelector
printf("%s 转发消息至BackUpClass\n",__func__);
return [[BackUpClass alloc] init];
}
// 返回self或者nil,则说明没有可以响应的目标,则进入下一步methodSignatureForSelector。
return nil;
}
第三步 常规转发阶段 | 慢速查找
获得一个方法签名。签名由一个能响应aSelector的实例生成。
有签名则进入消息转发的最后一步forwardInvocation。
/**
常规转发阶段 | 慢速查找
返回一个方法签名。签名由一个能响应aSelector的实例生成。
*/
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
{
NSString *selectorString = NSStringFromSelector(aSelector);
printf("%s %s\n", __func__, selectorString.UTF8String);
BackUpClass * backUp = [BackUpClass new];
NSMethodSignature * sign = [backUp methodSignatureForSelector:aSelector];
//有签名则进入消息转发的最后一步forwardInvocation
return sign;
}
也可以什么都不处理,至此本次消息转发结束,也不会crash。
/**
将sel转发给一个真正实现了sel的对象
也可以什么都不处理,至此本次消息转发结束,也不会crash。
*/
- (void)forwardInvocation:(NSInvocation *)anInvocation
{
printf("%s %s\n",__func__ , anInvocation.description.UTF8String);
// 创建备用消息接收对象
BackUpClass * backUp = [[BackUpClass alloc] init];
printf("%s 转发消息至BackUpClass\n",__func__);
[anInvocation invokeWithTarget:backUp];
}
写在后面
浅谈。可以交流。