据说NSProxy是少有的不继承NSObject的类,所以它应该是一个别致的存在。
关于NSProxy可参考官方文档,NSProxy的接口很少,但是感觉他的用处很多也很关键。
下面是对官方文档的翻译,英语翻译为中文,很蹩脚,感觉,领会精神吧!
NSProxy是一个为对象定义接口的抽象父类,并且为其它对象或者一些不存在的对象扮演了替身角色。通常,给proxy的消息被转发给实际对象或者导致proxy加载(转化它为)实际对象。NSProxy的子类能被用来实现透明的分布式消息(例如:NSDistantObject)或者延缓要花费昂贵代价创建的对象的实现。
概述
NSProxy实现被根类要求的基础方法,包括定义NSObject协议。然而,作为抽象类,他不实现初始化方法,并且会在收到任何他不响应的消息时引发异常。因此具体子类必须提供一个初始化或者创建方法并且重写forwardInvocation:和methodSignatureForSelector:方法来操纵它本身不执行的消息。不管怎样,一个子类的forwardInvocation:的实现应该需要处理invocation,例如通过网络转发invocation或者加载实际的对象并且把它传递给invocation.methodSignatureForSelector:被要求提供被给消息的参数类型信息;子类的实现应该考虑消息中参数类型,它需要据此促进和创建NSMethodSignature对象。参看NSDistantObject,NSInvocation和NSMethodSignature类指定的更多信息。
场景一:
场景一是取自开发文档的代码示例,他其实是通过消息传递机制实现了多重继承功能,使proxy拥有了NSSting与NSArray俩个类的方法属性
场景二:
场景二是解决了NSTimer轮播功能的循环引用的问题。其实我以前没有太注意这个问题,以后可以引以为戒
场景三:
就是如何实现偷天换日的功能,比如,如何让人像汽车一样run
#import
@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
@interface Car : NSObject
- (void)run;
@end
#import "Car.h"
@implementation Car
- (void)run{
NSLog(@"Car is running!");
}
@end
以上的例子都是通过methodSignatureForSelector:和forwardInvocation:实现消息转发来达到目的