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;
}

以上为笔者对于伪多继承实现的一些见解,如文中有任何错误请及时指正。

Demo地址

参考

iOS理解Objective-C中消息转发机制附Demo

上一篇下一篇

猜你喜欢

热点阅读