消息转发是解决object-c不能用多继承的一种方法,也是动态方法的一种。
先说下他的原理:后面再上代码解释。
消息转发其实就是当实例(类)不能处理一个消息时,在抛出异常前,系统给你一个方法你处理这个情况。
Foo * foo = [[Foo alloc] init];
[foo Bar];
[foo MissMethod];
可是在Foo类里面并没有实现MissMethod这个方法。
于是它会调用以下其中一个方法:
+(BOOL)resolveInstanceMethod:(SEL)name
{
NSLog(@" >> Instance resolving %@", NSStringFromSelector(name));
if (name == @selector(MissMethod)) {
class_addMethod([self class], name, (IMP)dynamicMethodIMP, "v@:");
return YES;
}
return [super resolveInstanceMethod:name];
}
+(BOOL)resolveClassMethod:(SEL)name
{
NSLog(@" >> Class resolving %@", NSStringFromSelector(name));
return [super resolveClassMethod:name];
}
它表示 动态加入dynamicMethodIMP这个方法。具体实现步骤为
1,首先判断是否实现了 resolveInstanceMethod,如果没有实现,返回 NULL,进入下一步处理;
2,如果实现了,调用 resolveInstanceMethod,获取返回值;
3,如果返回值为 YES,表示 resolveInstanceMethod 声称它已经提供了 selector 的实现,因此再次查找 method list,如果依然找到对应的 IMP,则返回该实现,否则提示警告信息,返回 NULL,进入下一步处理;
4,如果返回值为 NO,返回 NULL,进入下一步处理;
- (void)forwardInvocation:(NSInvocation *)anInvocation
{
SEL name = [anInvocation selector];
NSLog(@" >> forwardInvocation for selector %@", NSStringFromSelector(name));
Proxy * proxy = [[Proxy alloc] init] ;
if ([proxy respondsToSelector:name]) {
[anInvocation invokeWithTarget:proxy];
}
else {
[super forwardInvocation:anInvocation];
}
}
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
return [Proxy instanceMethodSignatureForSelector:aSelector];
}
最后会调用Proxy里的方法。