Objective C语言把能在运行期做的事情就推迟到运行期再决定。这就意味着,Objective C不仅需要一个编译器,而且需要一个运行期环境。这个运行期环境就是Runtime。而消息机制也是位于<objc/message.h>。
1、如果我们调用一个对象只有声明了方法并没有对其实现的话编译器在运行时会报错为
-[animal wight]: unrecognized selector sent to instance 0x1345809a0
2017-11-07 13:49:52.185 method[2076:1036713] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[animal wight]: unrecognized selector sent to instance 0x1345809a0'
*** First throw call stack:
消息转发机制一般会有三个过程
一、第一过程
+ (BOOL)resolveInstanceMethod:(SEL)sel(对象方法没有实现)
+ (BOOL)resolveClassMethod:(SEL)sel (类方法没有实现)
二、第二过程
- (id)forwardingTargetForSelector:(SEL)aSelector
三 、第三过程
- (NSMethodSignature*)methodSignatureForSelector:(SEL)aSelector
-(void)forwardInvocation:(NSInvocation*)anInvocation
1、如果我们在第一过程中动态添加了一个方法的实现了过程二和过程三就不会在执行、如果在第二过程做了处理 第三过程也不会在执行。如果过程一、二、三、都没有处理那么程序就会抛出异常。
下图是整个过程
2、如果我们在resolveInstanceMethod:(SEL)sel在这个方法中做了处理会怎么样呢?可以发现程序没有再抛出异常了 而是调用了我们增加的方法实现 解释一下这几个字段的意思可以发现仅仅调用了方法一, 二、三、四都没有调用。
关于生成签名的类型"v@:"解释一下。每一个方法会默认隐藏两个参数,self、_cmd,self代表方法调用者,_cmd代表这个方法的SEL,签名类型就是用来描述这个方法的返回值、参数的,v代表返回值为void,@表示self,:表示_cmd。
3、如果我们在方法一中没有做处理而是在方法二中做了处理其中的 car这个类中我在其的.m中实现了eat这个方法,就会将这个消息抛给car这个类中去寻找实现eat的方法。
4、如果- (id)forwardingTargetForSelector:(SEL)aSelector这个方法没有做处理
methodSignatureForSelector和forwardInvocationmethodSignatureForSelector用来生成方法签名,这个签名就是给forwardInvocation中的参数NSInvocation调用的。
开头我们要找的错误unrecognized selector sent to instance原因,原来就是因为methodSignatureForSelector这个方法中,由于没有找到run对应的实现方法,所以返回了一个空的方法签名,最终导致程序报错崩溃。
所以我们需要做的是自己新建方法签名,再在forwardInvocation中用你要转发的那个对象调用这个对应的签名,这样也实现了消息转发。
其中Delegate中实现了eat这个方法 。