方案一:
判断是否能够相应事件
+ (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 <objc/runtime.h>
@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