NSProxy实现伪多继承
2018-03-02 本文已影响0人
seej
Objective-C是不支持多继承的,但是基于消息转发机制我们可以使用NSProxy来实现伪多继承。
1.关于NSProxy
NSProxy是和NSObject同级的一个类,它只实现了<NSObject>的协议。
基于Objective-C的消息转发机制(iOS理解Objective-C中消息转发机制附Demo),在我们给NSProxy对象发送消息时,首先会根据selector在本类以及父类的方法列表中查找,如果找不到,则会启动消息转发机制。在一系列的消息转发过程中,如果 + resolveInstanceMethod: 以及 - forwardingTargetForSelector: 都未能对消息进行处理, - methodSignatureForSelector: 会被调用以获取方法签名,如果该方法返回了方法签名,则我们会在 - forwardInvocation: 方法中拿到相应消息的信息。
2.实现
首先我们需要创建一个继承自NSProxy的类并提供方法传入若干对象。
+ (instancetype)proxyWithObjs:(id)obj, ... NS_REQUIRES_NIL_TERMINATION {
NSMutableArray * objs = [NSMutableArray arrayWithObject:obj];
if (obj) {
va_list args;
va_start(args, obj);
id obj;
while ((obj = va_arg(args, id))) {
[objs addObject:obj];
}
va_end(args);
}
SEEProxy * instance = [SEEProxy alloc];
instance -> _objs = objs.copy;
return instance;
}
当消息转发进行到 - methodSignatureForSelector: 时我们需要在当前拥有的对象中查找方法,找到后将方法签名返回,代码如下:
- (NSMethodSignature *)methodSignatureForSelector:(SEL)sel {
__block NSMethodSignature * signature;
[_objs enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
//判断对象是否能够响应方法
if ([obj respondsToSelector:sel]) {
signature = [obj methodSignatureForSelector:sel];
*stop = YES;
}
}];
return signature;
}
返回方法签名后我们需要在 - forwardInvocation: 中对消息进行处理:
- (void)forwardInvocation:(NSInvocation *)invocation {
[_objs enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
//判断对象是否能够响应方法
if ([obj respondsToSelector:invocation.selector]) {
[invocation invokeWithTarget:obj];
*stop = YES;
}
}];
}
我们需要通过 - performSelector: 调用方法,为了防止崩溃在调用前最好先使用 - respondsToSelector: 判断是否能够响应方法,因此我们还需要实现 - respondsToSelector: 方法:
- (BOOL)respondsToSelector:(SEL)aSelector {
__block BOOL flag = [super respondsToSelector:aSelector];
if (flag) return flag;
[_objs enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
flag = [obj respondsToSelector:aSelector];
*stop = flag;
}];
return flag;
}
拓展:另外一种实现伪多继承的方式
上文中我们是在消息转发过程中的第三步进行操作来实现伪多继承,纵观整个消息转发过程,我们在第二步同样有操作的空间,代码如下:
- (id)forwardingTargetForSelector:(SEL)aSelector {
__block id target;
[_objs enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
//判断对象是否能够响应方法
if ([obj respondsToSelector:aSelector]) {
target = obj;
*stop = YES;
}
}];
return target;
}
以上为笔者对于伪多继承实现的一些见解,如文中有任何错误请及时指正。