今天去YY面试,问起了消息转发,竟然一时答不出来。
现在把iOS消息转发的流程过一遍。
首先我们要知道消息转发都有哪些方法以及其的调用流程,才能更好的掌握。
// 比如调用的方法没有实现或者不存在的时候OC会让你有机会选择哪个方法继续执行
//实例方法先调用:
+ (BOOL)resolveInstanceMethod:(SEL)sel
//类方法不存在时先调用
+ (BOOL)resolveClassMethod:(SEL)sel;
如果在以上的方法都没有做处理,OC还会给你第二个机会
-(id)forwardingTargetForSelector:(SEL)aSelector;
如果在第二次机会还没有处理,还有最后一次机会
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector;
- (void)forwardInvocation:(NSInvocation *)anInvocation;
举个栗子:
现在准备两个对象分别是Dog
和Pig
//Dog.h
#import <Foundation/Foundation.h>
@interface Dog : NSObject
-(void)eat;
+(void)sleep;
@end
#import "Dog.h"
@implementation Dog
@end
此处Dog.m
是没有实现-eat
和+sleep
方法的
调用eat
方法时会报一下错误
2018-03-06 00:09:06.337035+0800 MassageForward[2059:18708264] -[Dog eat]: unrecognized selector sent to instance 0x100513f10
2018-03-06 00:09:06.338884+0800 MassageForward[2059:18708264] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[Dog eat]: unrecognized selector sent to instance 0x100513f10'
现在我们在Don.m加上最开始的那几个方法,变成了以下样子
void eatSomthing(id self, SEL sel)
{
NSLog(@"%@ %s 【eat something】", self, sel_getName(sel));
}
// 决定选择哪个实例方法
+ (BOOL)resolveInstanceMethod:(SEL)sel
{
NSLog(@"%s %@",__func__,NSStringFromSelector(sel));
// 动态添加一个方法,如果不添加跳到Step 2
// if(sel == @selector(eat))
// {
// // 添加一个代替eat实现的方法
// class_addMethod(self, sel, (IMP)eatSomthing, "v@:");
// return YES;
// }
return [super resolveInstanceMethod:sel];
}
//当类方法不存在时调用 如: +eat()没有实现的话
+ (BOOL)resolveClassMethod:(SEL)sel
{
NSLog(@"%s %s",__func__, sel_getName(sel));
return [super resolveClassMethod:sel];
}
//=============================================Step 2=========================================================
-(id)forwardingTargetForSelector:(SEL)aSelector
{
NSLog(@"%s %@",__func__,NSStringFromSelector(aSelector));
//转发给Pig的实例,调用pig的eat方法,如果不转发则跳到Step 3
// return [Pig new];
return [super forwardingTargetForSelector:aSelector];
}
//=============================================Step 3=========================================================
// 签名
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
{
NSLog(@"%s %@",__func__,NSStringFromSelector(aSelector));
if(aSelector == @selector(eat))
{
return [NSMethodSignature signatureWithObjCTypes:"v@:"];
}
return [super methodSignatureForSelector:aSelector];
}
- (void)forwardInvocation:(NSInvocation *)anInvocation
{
NSLog(@"%s",__func__);
SEL selector = [anInvocation selector];
Pig *pig = [Pig new];
if([pig respondsToSelector:selector])
{
[anInvocation invokeWithTarget:pig];
}
}
这样子,就可以实现,在没有实现eat
或者sleep
方法时,实现了消息转发,从而不会导致程序崩溃。
要说明的是"v@:"
,每个方法会有两个默认值,一个是self
和_cmd
, 表示方法的拥有者和SEL, 签名类型就是描述这个方法的参数和返回值的
其中v
表示void
, @
表示self
,:
表示_cmd
此示例中就是Dog
调用eat
方法后被Pig
的eat
方法接收了。这就是OC中的消息转发。