消息转发
![这里写图片描述] WX20180910-180214@2x.png消息转发步骤
- 调用resolveInstanceMethod:方法 (或 resolveClassMethod:)。允许用户在此时为该 Class 动态添加实现。如果有实现了,则调用并返回YES,那么重新开始objc_msgSend流程。这一次对象会响应这个选择器,一般是因为它已经调用过 class_addMethod。如果仍没实现,继续下面的动作。
- 调用forwardingTargetForSelector:方法,尝试找到一个能响应该消息的对象。如果获取到,则直接把消息转发给它,返回非 nil 对象。否则返回 nil ,继续下面的动作。注意,这里不要返回 self ,否则会形成死循环。
- 调用methodSignatureForSelector:方法,尝试获得一个方法签名。如果获取不到,则直接调用doesNotRecognizeSelector抛出异常。如果能获取,则返回非nil:创建一个 NSlnvocation 并传给forwardInvocation:。
- 调用forwardInvocation:方法,将第3步获取到的方法签名包装成 Invocation 传入,如何处理就在这里面了,并返回非ni。
- 调用doesNotRecognizeSelector: ,默认的实现是抛出异常。如果第3步没能获得一个方法签名,执行该步骤。
上面前4个方法均是模板方法,开发者可以override,由 runtime 来调用。最常见的实现消息转发:就是重写方法3和4,吞掉一个消息或者代理给其他对象都是没问题的
也就是说_objc_msgForward在进行消息转发的过程中会涉及以下这几个方法:
resolveInstanceMethod:方法 (或 resolveClassMethod:)。
forwardingTargetForSelector:方法
methodSignatureForSelector:方法
forwardInvocation:方法
doesNotRecognizeSelector: 方法
NSProxy
概述
NSProxy是一个为对象定义接口的抽象父类,并且为其它对象或者一些不存在的对象扮演了替身角色。通常,给proxy的消息被转发给实际对象或者导致proxy加载(转化它为)实际对象。NSProxy的子类能被用来实现透明的分布式消息(例如:NSDistantObject)或者延缓要花费昂贵代价创建的对象的实现。
主要是利用消息转发机制
声明
![这里写图片描述] WX20180910-113141@2x.png没有init方法,直接alloc。并且可以看到,它遵守了 NSObject 协议,并且第一个 Ivar 是一个 isa 指针,因此它完全是可以拿来当一个 NSObject 或其派生类来使用的。
一般使用两个方法
// 生成方法签名
- (nullable NSMethodSignature *)methodSignatureForSelector:(SEL)sel
// 方法的签名配发
- (void)forwardInvocation:(NSInvocation *)invocation
三个例子
取自开发文档的代码示例,他其实是通过消息传递机制实现了多重继承功能,使proxy拥有了NSSting与NSArray俩个类的方法属性
解决了NSTimer轮播功能的循环引用的问题。其实我以前没有太注意这个问题,以后可以引以为戒
例子三
就是如何实现偷天换日的功能,比如,如何让人像汽车一样run
#import <Foundation/Foundation.h>
@interface Person : NSObject
- (void)run;
@end
#import "Person.h"
#import "Car.h"
@implementation Person
//返回和参数的类型信息
- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector{
NSString *sel = NSStringFromSelector(aSelector);
if ([sel isEqualToString:@"run"]) {
return [NSMethodSNSInvocationignature signatureWithObjCTypes:"v@:"];
}
return [super methodSignatureForSelector:aSelector];
//引发NSInvalidArgumentException。在你具体子类中重写这个方法,为被给选择器和你的代理对象代表的类返回合适的NSMethodSignature对象
}
//加载对象,把对象传递给anInvocation
- (void)forwardInvocation:(NSInvocation *)anInvocation{
SEL selector = [anInvocation selector];
Car *car = [[Car alloc] init];
if ([car respondsToSelector:selector]) {
[anInvocation invokeWithTarget:car];//传递一个invocation给proxy代表的真的对象
}
}
#import <Foundation/Foundation.h>
@interface Car : NSObject
- (void)run;
@end
#import "Car.h"
@implementation Car
- (void)run{
NSLog(@"Car is running!");
}
@end
以上的例子都是通过methodSignatureForSelector:和forwardInvocation:实现消息转发来达到目的
网友评论