当你初次进入NSProxy文档的时候你会发现,这里面方法少的可怜,甚至吝啬到连init方法都木有!
NS_ROOT_CLASS @interface NSProxy <NSObject> { Class isa; }
+ (id)alloc;
+ (id)allocWithZone:(nullable NSZone *)zone ;
+ (Class)class;
- (void)forwardInvocation:(NSInvocation *)invocation;
- (nullable NSMethodSignature *)methodSignatureForSelector: (SEL)sel
为什么说是扫地僧呢,因为他可拦截任何方法的实现,即使.h方法都没有声明的方法,下面直奔主题。
Q:假如我有个CQTeacher类,.m里面有个方法teach:,如何在未申明的时候就可以调用,并且如何实现调用过程中拦截改变其参数???。
- 创建一个自定义的类CQProxy继承NSProxy在.h中添加一个方法用来将如何传入的对象,进行转换。
@interface CQProxy : NSProxy
-(void)transformObjc:(NSObject *)objc;
@end
- .m文件
//0.外界传入的objc -(void)transformObjc:(NSObject *)objc{ self.objc = objc; }
//1.查询该方法的方法签名,自己没有就去super中去找 -(NSMethodSignature *)methodSignatureForSelector:(SEL)sel{ NSMethodSignature *signature = nil; //找方法签名 if ([self.objc methodSignatureForSelector:sel]) { signature = [self.objc methodSignatureForSelector:sel]; } else{ signature = [super methodSignatureForSelector:sel]; } return signature; }
//2.调用方法实现
-(void)forwardInvocation:(NSInvocation *)invocation{ if(self.objc){ //设置想要的目标类 [invocation setTarget:self.objc]; //判断定是否是我们想要的类型 if ([self.objc isKindOfClass:[NSClassFromString(@"CQTeacher") class]]) { //这里,就是拦截参数,设置成我们想要的参数,即不管外界传的什么参数,我们传入的参数永远是iOS NSString *str = @"iOS"; //下标 0,1都被占用了。 我们设置参数必须从2开始 [invocation setArgument:&str atIndex:2]; } //开始调用 [invocation invoke]; } }
看完自定义的CQProxy实现后,我们就可以来测试下是否可调用 拦截,我们在控制器中创建CQTeacher类的对象,然后进行转换
CQTeacher *teacher = [[CQTeacher alloc] init]; //没有init方法 只需要alloc即可。 CQProxy *proxy = [CQProxy alloc]; //转换类 [proxy transformObjc:teacher]; //即使我们传入的是java打印的却是iOS //即使我们并没在好.h申明方法,依旧可以调用 [proxy performSelector:@selector(teach:) withObject:@"java"];